import os import tarfile import tempfile from pathlib import Path import pytest import requests from .support import PyScriptTest, with_execution_thread @pytest.fixture def pyodide_0_22_0_tar(request): """ Fixture which returns a local copy of pyodide. It uses pytest-cache to avoid re-downloading it between runs. """ URL = "https://github.com/pyodide/pyodide/releases/download/0.22.0/pyodide-core-0.22.0.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(self): self.pyscript_run( """ name = "foobar" import js config = js.pyscript_get_config() js.console.log("config name:", config.name) """ ) assert self.console.log.lines[-1] == "config name: foobar" def test_py_config_external(self): pyconfig_toml = """ name = "app with external config" """ self.writefile("pyconfig.toml", pyconfig_toml) self.pyscript_run( """ import js config = js.pyscript_get_config() js.console.log("config name:", config.name) """ ) 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. def test_interpreter_config(self, pyodide_0_22_0_tar): unzip(pyodide_0_22_0_tar, extract_to=self.tmpdir) self.pyscript_run( """ { "interpreters": [{ "src": "/pyodide/pyodide.js", "name": "my-own-pyodide", "lang": "python" }] } import sys, js pyodide_version = sys.modules["pyodide"].__version__ js.console.log("version", pyodide_version) display(pyodide_version) """, ) assert self.console.log.lines[-1] == "version 0.22.0" version = self.page.locator("py-script").inner_text() assert version == "0.22.0" def test_runtime_still_works_but_shows_deprecation_warning( self, pyodide_0_22_0_tar ): unzip(pyodide_0_22_0_tar, extract_to=self.tmpdir) self.pyscript_run( """ { "runtimes": [{ "src": "/pyodide/pyodide.js", "name": "my-own-pyodide", "lang": "python" }] } import sys, js pyodide_version = sys.modules["pyodide"].__version__ js.console.log("version", pyodide_version) display(pyodide_version) """, ) assert self.console.log.lines[-1] == "version 0.22.0" version = self.page.locator("py-script").inner_text() assert version == "0.22.0" deprecation_banner = self.page.wait_for_selector(".alert-banner") expected_message = ( "The configuration option `config.runtimes` is deprecated. " "Please use `config.interpreters` instead." ) assert deprecation_banner.inner_text() == expected_message 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 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 def test_multiple_py_config(self): self.pyscript_run( """ name = "foobar" this is ignored and won't even be parsed import js config = js.pyscript_get_config() js.console.log("config name:", config.name) """ ) 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 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" ) 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" } ] } import js js.console.log("hello world"); """ 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"] import js import a, b js.console.log(a.x) js.console.log(b.x) """ ) assert self.console.log.lines[-2:] == [ "hello from A", "hello from B", ] 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"] import js from pkg.a import x js.console.log(x) """ ) assert self.console.log.lines[-1] == "hello from A"