import os import tarfile import tempfile from pathlib import Path import pytest import requests from .support import PyScriptTest, with_execution_thread PYODIDE_VERSION = "0.23.4" @pytest.fixture def pyodide_tar(request): """ Fixture which returns a local copy of pyodide. It uses pytest-cache to avoid re-downloading it between runs. """ URL = ( f"https://github.com/pyodide/pyodide/releases/download/{PYODIDE_VERSION}/" f"pyodide-core-{PYODIDE_VERSION}.tar.bz2" ) tar_name = Path(URL).name val = request.config.cache.get(tar_name, None) if val is None: response = requests.get(URL, stream=True) TMP_DIR = tempfile.mkdtemp() TMP_TAR_LOCATION = os.path.join(TMP_DIR, tar_name) with open(TMP_TAR_LOCATION, "wb") as f: f.write(response.raw.read()) val = TMP_TAR_LOCATION request.config.cache.set(tar_name, val) return val def unzip(location, extract_to="."): file = tarfile.open(name=location, mode="r:bz2") file.extractall(path=extract_to) # Disable the main/worker dual testing, for two reasons: # # 1. the logic happens before we start the worker, so there is # no point in running these tests twice # # 2. the logic to inject execution_thread into works only with # plain tags, but here we want to test all weird combinations # of config @with_execution_thread(None) class TestConfig(PyScriptTest): def test_py_config_inline_pyscript(self): self.pyscript_run( """ name = "foobar" from pyscript import window, document promise = await document.currentScript._pyodide.promise window.console.log("config name:", promise.config.name) """ ) assert self.console.log.lines[-1] == "config name: foobar" @pytest.mark.skip("ERROR_SCRIPT: works with not with """ ) assert self.console.log.lines[-1] == "config name: foobar" @pytest.mark.skip("ERROR_SCRIPT: works with not with """ ) assert self.console.log.lines[-1] == "config name: app with external config" # The default pyodide version is newer than # the one we are loading below (after downloading locally) # which is 0.22.0 # The test checks if loading a different interpreter is possible # and that too from a locally downloaded file without needing # the use of explicit `indexURL` calculation. @pytest.mark.skip("Interpreters key is not implemented in PyScript Next") def test_interpreter_config(self, pyodide_tar): unzip(pyodide_tar, extract_to=self.tmpdir) self.pyscript_run( """ { "interpreters": [{ "src": "/pyodide/pyodide.js", "name": "my-own-pyodide", "lang": "python" }] } """, ) assert self.console.log.lines[-1] == f"version {PYODIDE_VERSION}" @pytest.mark.skip("FIXME: We need to restore the banner.") def test_invalid_json_config(self): # we need wait_for_pyscript=False because we bail out very soon, # before being able to write 'PyScript page fully initialized' self.pyscript_run( """ [[ """, wait_for_pyscript=False, ) banner = self.page.wait_for_selector(".py-error") assert "SyntaxError: Unexpected end of JSON input" in self.console.error.text expected = ( "(PY1000): The config supplied: [[ is an invalid JSON and cannot be " "parsed: SyntaxError: Unexpected end of JSON input" ) assert banner.inner_text() == expected @pytest.mark.skip("FIXME: We need to restore the banner.") def test_invalid_toml_config(self): # we need wait_for_pyscript=False because we bail out very soon, # before being able to write 'PyScript page fully initialized' self.pyscript_run( """ [[ """, wait_for_pyscript=False, ) banner = self.page.wait_for_selector(".py-error") assert "SyntaxError: Expected DoubleQuote" in self.console.error.text expected = ( "(PY1000): The config supplied: [[ is an invalid TOML and cannot be parsed: " "SyntaxError: Expected DoubleQuote, Whitespace, or [a-z], [A-Z], " '[0-9], "-", "_" but "\\n" found.' ) assert banner.inner_text() == expected @pytest.mark.skip("FIXME: We need to restore the banner.") def test_multiple_py_config(self): self.pyscript_run( """ name = "foobar" this is ignored and won't even be parsed """ ) banner = self.page.wait_for_selector(".py-warning") expected = ( "Multiple tags detected. Only the first " "is going to be parsed, all the others will be ignored" ) assert banner.text_content() == expected @pytest.mark.skip("Interpreters key is not implemented in PyScript Next") def test_no_interpreter(self): snippet = """ { "interpreters": [] } """ self.pyscript_run(snippet, wait_for_pyscript=False) div = self.page.wait_for_selector(".py-error") assert ( div.text_content() == "(PY1000): Fatal error: config.interpreter is empty" ) @pytest.mark.skip("Interpreters key is not implemented in PyScript Next") def test_multiple_interpreter(self): snippet = """ { "interpreters": [ { "src": "https://cdn.jsdelivr.net/pyodide/v0.23.2/full/pyodide.js", "name": "pyodide-0.23.2", "lang": "python" }, { "src": "http://...", "name": "this will be ignored", "lang": "this as well" } ] } """ self.pyscript_run(snippet) banner = self.page.wait_for_selector(".py-warning") expected = ( "Multiple interpreters are not supported yet.Only the first will be used" ) assert banner.text_content() == expected assert self.console.log.lines[-1] == "hello world" def test_paths(self): self.writefile("a.py", "x = 'hello from A'") self.writefile("b.py", "x = 'hello from B'") self.pyscript_run( """ [[fetch]] files = ["./a.py", "./b.py"] """ ) assert self.console.log.lines[-2:] == [ "hello from A", "hello from B", ] @pytest.mark.skip("FIXME: We need to restore the banner.") def test_paths_that_do_not_exist(self): self.pyscript_run( """ [[fetch]] files = ["./f.py"] """, wait_for_pyscript=False, ) expected = "(PY0404): Fetching from URL ./f.py failed with " "error 404" inner_html = self.page.locator(".py-error").inner_html() assert expected in inner_html assert expected in self.console.error.lines[-1] def test_paths_from_packages(self): self.writefile("utils/__init__.py", "") self.writefile("utils/a.py", "x = 'hello from A'") self.pyscript_run( """ [[fetch]] from = "utils" to_folder = "pkg" files = ["__init__.py", "a.py"] """ ) assert self.console.log.lines[-1] == "hello from A"