mirror of
https://github.com/pyscript/pyscript.git
synced 2025-12-20 10:47:35 -05:00
352 lines
12 KiB
Python
352 lines
12 KiB
Python
import re
|
|
|
|
import pytest
|
|
|
|
from .support import JsErrors, PyScriptTest
|
|
|
|
|
|
class TestBasic(PyScriptTest):
|
|
def test_pyscript_hello(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>
|
|
print('hello pyscript')
|
|
</py-script>
|
|
"""
|
|
)
|
|
assert self.console.log.lines[-1] == "hello pyscript"
|
|
|
|
def test_python_exception(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>
|
|
print('hello pyscript')
|
|
raise Exception('this is an error')
|
|
</py-script>
|
|
"""
|
|
)
|
|
assert "hello pyscript" in self.console.log.lines
|
|
# check that we sent the traceback to the console
|
|
tb_lines = self.console.error.lines[-1].splitlines()
|
|
assert tb_lines[0] == "[pyexec] Python exception:"
|
|
assert tb_lines[1] == "Traceback (most recent call last):"
|
|
assert tb_lines[-1] == "Exception: this is an error"
|
|
#
|
|
# check that we show the traceback in the page. Note that here we
|
|
# display the "raw" python traceback, without the "[pyexec] Python
|
|
# exception:" line (which is useful in the console, but not for the
|
|
# user)
|
|
pre = self.page.locator("py-script > pre")
|
|
tb_lines = pre.inner_text().splitlines()
|
|
assert tb_lines[0] == "Traceback (most recent call last):"
|
|
assert tb_lines[-1] == "Exception: this is an error"
|
|
|
|
def test_python_exception_in_event_handler(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<button py-click="onclick()">Click me</button>
|
|
<py-script>
|
|
def onclick():
|
|
raise Exception("this is an error inside handler")
|
|
</py-script>
|
|
"""
|
|
)
|
|
|
|
self.page.locator("button").click()
|
|
|
|
## error in console
|
|
tb_lines = self.console.error.lines[-1].splitlines()
|
|
assert tb_lines[0] == "[pyexec] Python exception:"
|
|
assert tb_lines[1] == "Traceback (most recent call last):"
|
|
assert tb_lines[-1] == "Exception: this is an error inside handler"
|
|
|
|
## error in DOM
|
|
tb_lines = self.page.locator(".py-error").inner_text().splitlines()
|
|
assert tb_lines[0] == "Traceback (most recent call last):"
|
|
assert tb_lines[-1] == "Exception: this is an error inside handler"
|
|
|
|
def test_execution_in_order(self):
|
|
"""
|
|
Check that they py-script tags are executed in the same order they are
|
|
defined
|
|
"""
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>import js; js.console.log('one')</py-script>
|
|
<py-script>js.console.log('two')</py-script>
|
|
<py-script>js.console.log('three')</py-script>
|
|
<py-script>js.console.log('four')</py-script>
|
|
"""
|
|
)
|
|
assert self.console.log.lines[-4:] == [
|
|
"one",
|
|
"two",
|
|
"three",
|
|
"four",
|
|
]
|
|
|
|
def test_escaping_of_angle_brackets(self):
|
|
"""
|
|
Check that py-script tags escape angle brackets
|
|
"""
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>import js; js.console.log(1<2, 1>2)</py-script>
|
|
<py-script>js.console.log("<div></div>")</py-script>
|
|
"""
|
|
)
|
|
|
|
assert self.console.log.lines[-2:] == ["true false", "<div></div>"]
|
|
|
|
def test_packages(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-config>
|
|
# we use asciitree because it's one of the smallest packages
|
|
# which are built and distributed with pyodide
|
|
packages = ["asciitree"]
|
|
</py-config>
|
|
<py-script>
|
|
import js
|
|
import asciitree
|
|
js.console.log('hello', asciitree.__name__)
|
|
</py-script>
|
|
"""
|
|
)
|
|
|
|
assert self.console.log.lines[-3:] == [
|
|
"Loading asciitree", # printed by pyodide
|
|
"Loaded asciitree", # printed by pyodide
|
|
"hello asciitree", # printed by us
|
|
]
|
|
|
|
def test_non_existent_package(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-config>
|
|
packages = ["nonexistendright"]
|
|
</py-config>
|
|
""",
|
|
wait_for_pyscript=False,
|
|
)
|
|
|
|
expected_alert_banner_msg = (
|
|
"(PY1001): Unable to install package(s) 'nonexistendright'. "
|
|
"Unable to find package in PyPI. Please make sure you have "
|
|
"entered a correct package name."
|
|
)
|
|
|
|
alert_banner = self.page.wait_for_selector(".alert-banner")
|
|
assert expected_alert_banner_msg in alert_banner.inner_text()
|
|
|
|
def test_no_python_wheel(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-config>
|
|
packages = ["opsdroid"]
|
|
</py-config>
|
|
""",
|
|
wait_for_pyscript=False,
|
|
)
|
|
|
|
expected_alert_banner_msg = (
|
|
"(PY1001): Unable to install package(s) 'opsdroid'. "
|
|
"Reason: Can't find a pure Python 3 Wheel for package(s) 'opsdroid'"
|
|
)
|
|
|
|
alert_banner = self.page.wait_for_selector(".alert-banner")
|
|
assert expected_alert_banner_msg in alert_banner.inner_text()
|
|
|
|
def test_dynamically_add_py_script_tag(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<script>
|
|
function addPyScriptTag() {
|
|
let tag = document.createElement('py-script');
|
|
tag.innerHTML = "print('hello world')";
|
|
document.body.appendChild(tag);
|
|
}
|
|
</script>
|
|
<button onclick="addPyScriptTag()">Click me</button>
|
|
"""
|
|
)
|
|
self.page.locator("button").click()
|
|
|
|
self.page.wait_for_selector("py-terminal")
|
|
assert self.console.log.lines[-1] == "hello world"
|
|
|
|
def test_py_script_src_attribute(self):
|
|
self.writefile("foo.py", "print('hello from foo')")
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script src="foo.py"></py-script>
|
|
"""
|
|
)
|
|
assert self.console.log.lines[-1] == "hello from foo"
|
|
|
|
def test_py_script_src_not_found(self):
|
|
with pytest.raises(JsErrors) as exc:
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script src="foo.py"></py-script>
|
|
"""
|
|
)
|
|
assert "Failed to load resource" in self.console.error.lines[0]
|
|
|
|
error_msgs = str(exc.value)
|
|
|
|
expected_msg = "(PY0404): Fetching from URL foo.py failed with error 404"
|
|
|
|
assert expected_msg in error_msgs
|
|
assert self.assert_banner_message(expected_msg)
|
|
|
|
pyscript_tag = self.page.locator("py-script")
|
|
assert pyscript_tag.inner_html() == ""
|
|
|
|
def test_js_version(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>
|
|
</py-script>
|
|
"""
|
|
)
|
|
self.page.add_script_tag(content="console.log(pyscript.version)")
|
|
|
|
assert (
|
|
re.match(r"\d{4}\.\d{2}\.\d+(\.[a-zA-Z0-9]+)?", self.console.log.lines[-1])
|
|
is not None
|
|
)
|
|
|
|
def test_python_version(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>
|
|
import js
|
|
js.console.log(pyscript.__version__)
|
|
js.console.log(str(pyscript.version_info))
|
|
</py-script>
|
|
"""
|
|
)
|
|
assert (
|
|
re.match(r"\d{4}\.\d{2}\.\d+(\.[a-zA-Z0-9]+)?", self.console.log.lines[-2])
|
|
is not None
|
|
)
|
|
assert (
|
|
re.match(
|
|
r"version_info\(year=\d{4}, month=\d{2}, "
|
|
r"minor=\d+, releaselevel='([a-zA-Z0-9]+)?'\)",
|
|
self.console.log.lines[-1],
|
|
)
|
|
is not None
|
|
)
|
|
|
|
def test_assert_no_banners(self):
|
|
"""
|
|
Test that the DOM doesn't contain error/warning banners
|
|
"""
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>
|
|
import pyscript
|
|
pyscript.showWarning("hello")
|
|
pyscript.showWarning("world")
|
|
</py-script>
|
|
"""
|
|
)
|
|
with pytest.raises(AssertionError, match="Found 2 alert banners"):
|
|
self.assert_no_banners()
|
|
|
|
def test_deprecated_globals(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>
|
|
# trigger various warnings
|
|
create("div", classes="a b c")
|
|
assert sys.__name__ == 'sys'
|
|
dedent("")
|
|
format_mime("")
|
|
assert MIME_RENDERERS['text/html'] is not None
|
|
console.log("hello")
|
|
PyScript.loop
|
|
</py-script>
|
|
|
|
<div id="mydiv"></div>
|
|
"""
|
|
)
|
|
banner = self.page.locator(".py-warning")
|
|
messages = banner.all_inner_texts()
|
|
assert messages == [
|
|
"The PyScript object is deprecated. Please use pyscript instead.",
|
|
"Direct usage of console is deprecated. Please use js.console instead.",
|
|
"MIME_RENDERERS is deprecated. This is a private implementation detail of pyscript. You should not use it.", # noqa: E501
|
|
"format_mime is deprecated. This is a private implementation detail of pyscript. You should not use it.", # noqa: E501
|
|
"Direct usage of dedent is deprecated. Please use from textwrap import dedent instead.",
|
|
"Direct usage of sys is deprecated. Please use import sys instead.",
|
|
"Direct usage of create is deprecated. Please use pyscript.create instead.",
|
|
]
|
|
|
|
def test_getPySrc_returns_source_code(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<py-script>
|
|
print("hello world!")
|
|
</py-script>
|
|
"""
|
|
)
|
|
|
|
pyscript_tag = self.page.locator("py-script")
|
|
assert pyscript_tag.inner_html() == ""
|
|
assert (
|
|
pyscript_tag.evaluate("node => node.getPySrc()")
|
|
== 'print("hello world!")\n'
|
|
)
|
|
|
|
@pytest.mark.skip(reason="pys-onClick is broken, we should kill it, see #1213")
|
|
def test_pys_onClick_shows_deprecation_warning(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<button id="1" pys-onClick="myfunc()">Click me</button>
|
|
<py-script>
|
|
def myfunc():
|
|
print("hello world")
|
|
|
|
</py-script>
|
|
"""
|
|
)
|
|
banner = self.page.locator(".alert-banner")
|
|
expected_message = (
|
|
"The attribute 'pys-onClick' and 'pys-onKeyDown' are "
|
|
"deprecated. Please 'py-click=\"myFunction()\"' or "
|
|
"'py-keydown=\"myFunction()\"' instead."
|
|
)
|
|
assert banner.inner_text() == expected_message
|
|
|
|
def test_py_attribute_without_id(self):
|
|
self.pyscript_run(
|
|
"""
|
|
<button py-click="myfunc()">Click me</button>
|
|
<py-script>
|
|
def myfunc():
|
|
print("hello world!")
|
|
</py-script>
|
|
"""
|
|
)
|
|
btn = self.page.wait_for_selector("button")
|
|
btn.click()
|
|
assert self.console.log.lines[-1] == "hello world!"
|
|
assert self.console.error.lines == []
|
|
|
|
def test_py_mount_shows_deprecation_warning(self):
|
|
# last non-deprecated version: 2023.03.1
|
|
self.pyscript_run(
|
|
"""
|
|
<div id="foo" py-mount></div>
|
|
"""
|
|
)
|
|
banner = self.page.locator(".alert-banner")
|
|
expected_message = (
|
|
'The "py-mount" attribute is deprecated. '
|
|
+ "Please add references to HTML Elements manually in your script."
|
|
)
|
|
assert banner.inner_text() == expected_message
|