import pytest
from webdriver.bidi.modules.script import ContextTarget, RealmTarget, ScriptEvaluateResultException
from ... import any_int, any_string, recursive_compare
from .. import any_stack_trace
@pytest.mark.asyncio
async def test_sandbox(bidi_session, new_tab):
# Make changes in window
await bidi_session.script.evaluate(
expression="window.foo = 1",
target=ContextTarget(new_tab["context"]),
await_promise=True,
)
# Check that changes are not present in sandbox
result_in_sandbox = await bidi_session.script.evaluate(
expression="window.foo",
target=ContextTarget(new_tab["context"], "sandbox"),
await_promise=True,
)
assert result_in_sandbox == {"type": "undefined"}
# Make changes in sandbox
await bidi_session.script.evaluate(
expression="window.bar = 1",
target=ContextTarget(new_tab["context"], "sandbox"),
await_promise=True,
)
# Make sure that changes are present in sandbox
result_in_sandbox = await bidi_session.script.evaluate(
expression="window.bar",
target=ContextTarget(new_tab["context"], "sandbox"),
await_promise=True,
)
assert result_in_sandbox == {"type": "number", "value": 1}
# Make sure that changes didn't leak from sandbox
result = await bidi_session.script.evaluate(
expression="window.bar",
target=ContextTarget(new_tab["context"]),
await_promise=True,
)
assert result == {"type": "undefined"}
@pytest.mark.asyncio
async def test_sandbox_with_empty_name(bidi_session, new_tab):
# An empty string as a `sandbox` means the default realm should be used.
await bidi_session.script.evaluate(
expression="window.foo = 'bar'",
target=ContextTarget(new_tab["context"], ""),
await_promise=True,
)
# Make sure that we can find the sandbox with the empty name.
result = await bidi_session.script.evaluate(
expression="window.foo",
target=ContextTarget(new_tab["context"], ""),
await_promise=True,
)
assert result == {"type": "string", "value": "bar"}
# Make sure that we can find the value in the default realm.
result = await bidi_session.script.evaluate(
expression="window.foo",
target=ContextTarget(new_tab["context"]),
await_promise=True,
)
assert result == {"type": "string", "value": "bar"}
@pytest.mark.asyncio
async def test_switch_sandboxes(bidi_session, new_tab):
# Test that sandboxes are retained when switching between them
await bidi_session.script.evaluate(
expression="window.foo = 1",
target=ContextTarget(new_tab["context"], "sandbox_1"),
await_promise=True,
)
await bidi_session.script.evaluate(
expression="window.foo = 2",
target=ContextTarget(new_tab["context"], "sandbox_2"),
await_promise=True,
)
result_in_sandbox_1 = await bidi_session.script.evaluate(
expression="window.foo",
target=ContextTarget(new_tab["context"], "sandbox_1"),
await_promise=True,
)
assert result_in_sandbox_1 == {"type": "number", "value": 1}
result_in_sandbox_2 = await bidi_session.script.evaluate(
expression="window.foo",
target=ContextTarget(new_tab["context"], "sandbox_2"),
await_promise=True,
)
assert result_in_sandbox_2 == {"type": "number", "value": 2}
@pytest.mark.asyncio
async def test_sandbox_with_side_effects(bidi_session, new_tab):
# Make sure changing the node in sandbox will affect the other sandbox as well
await bidi_session.script.evaluate(
expression="document.querySelector('body').textContent = 'foo'",
target=ContextTarget(new_tab["context"], "sandbox_1"),
await_promise=True,
)
expected_value = {"type": "string", "value": "foo"}
result_in_sandbox_1 = await bidi_session.script.evaluate(
expression="document.querySelector('body').textContent",
target=ContextTarget(new_tab["context"], "sandbox_1"),
await_promise=True,
)
assert result_in_sandbox_1 == expected_value
result_in_sandbox_2 = await bidi_session.script.evaluate(
expression="document.querySelector('body').textContent",
target=ContextTarget(new_tab["context"], "sandbox_2"),
await_promise=True,
)
assert result_in_sandbox_2 == expected_value
@pytest.mark.asyncio
async def test_sandbox_returns_same_node(bidi_session, new_tab):
node = await bidi_session.script.evaluate(
expression="document.querySelector('body')",
target=ContextTarget(new_tab["context"]),
await_promise=True,
)
recursive_compare({"type": "node", "sharedId": any_string}, node)
node_sandbox = await bidi_session.script.evaluate(
expression="document.querySelector('body')",
target=ContextTarget(new_tab["context"], sandbox="sandbox_1"),
await_promise=True,
)
assert node_sandbox == node
@pytest.mark.asyncio
@pytest.mark.parametrize("await_promise", [True, False])
async def test_exception_details(bidi_session, new_tab, await_promise):
if await_promise:
expression = "Promise.reject(1)"
else:
expression = "throw 1"
with pytest.raises(ScriptEvaluateResultException) as exception:
await bidi_session.script.evaluate(
expression=expression,
target=ContextTarget(new_tab["context"], "sandbox"),
await_promise=await_promise,
)
recursive_compare(
{
"realm": any_string,
"exceptionDetails": {
"columnNumber": any_int,
"exception": {"type": "number", "value": 1},
"lineNumber": any_int,
"stackTrace": any_stack_trace,
"text": any_string,
},
},
exception.value.result,
)
@pytest.mark.asyncio
async def test_target_realm(bidi_session, top_context, default_realm):
result = await bidi_session.script.evaluate(
raw_result=True,
expression="window.foo = 3",
target=ContextTarget(top_context["context"], "sandbox"),
await_promise=True,
)
realm = result["realm"]
# Make sure that sandbox realm id is different from default
assert realm != default_realm
result = await bidi_session.script.evaluate(
raw_result=True,
expression="window.foo",
target=RealmTarget(realm),
await_promise=True,
)
recursive_compare(
{"realm": realm, "result": {"type": "number", "value": 3}}, result
)