import os from pathlib import Path import time import pytest def test_init(selenium_standalone): assert 'Python initialization complete' in selenium_standalone.logs assert len(selenium_standalone.driver.window_handles) == 1 def test_webbrowser(selenium): selenium.run("import antigravity") time.sleep(2) assert len(selenium.driver.window_handles) == 2 def test_print(selenium): selenium.run("print('This should be logged')") assert 'This should be logged' in selenium.logs def test_python2js(selenium): assert selenium.run_js('return pyodide.runPython("None") === undefined') assert selenium.run_js('return pyodide.runPython("True") === true') assert selenium.run_js('return pyodide.runPython("False") === false') assert selenium.run_js('return pyodide.runPython("42") === 42') assert selenium.run_js('return pyodide.runPython("3.14") === 3.14') # Need to test all three internal string representations in Python: UCS1, # UCS2 and UCS4 assert selenium.run_js( 'return pyodide.runPython("\'ascii\'") === "ascii"') assert selenium.run_js( 'return pyodide.runPython("\'ιωδιούχο\'") === "ιωδιούχο"') assert selenium.run_js( 'return pyodide.runPython("\'碘化物\'") === "碘化物"') assert selenium.run_js( 'let x = pyodide.runPython("b\'bytes\'");\n' 'return (x instanceof window.Uint8ClampedArray) && ' '(x.length === 5) && ' '(x[0] === 98)') assert selenium.run_js( 'let x = pyodide.runPython("[1, 2, 3]");\n' 'return (x instanceof window.Array) && (x.length === 3) && ' '(x[0] == 1) && (x[1] == 2) && (x[2] == 3)') assert selenium.run_js( 'let x = pyodide.runPython("{42: 64}");\n' 'return (typeof x === "object") && ' '(x[42] === 64)') assert selenium.run_js( 'let x = pyodide.runPython("open(\'/foo.txt\', \'wb\')")\n' 'return (x.tell() === 0)\n') def test_pythonexc2js(selenium): try: selenium.run_js('return pyodide.runPython("5 / 0")') except selenium.JavascriptException as e: assert('ZeroDivisionError' in str(e)) else: assert False, 'Expected exception' def test_js2python(selenium): selenium.run_js( 'window.jsstring = "碘化物";\n' 'window.jsnumber0 = 42;\n' 'window.jsnumber1 = 42.5;\n' 'window.jsundefined = undefined;\n' 'window.jsnull = null;\n' 'window.jstrue = true;\n' 'window.jsfalse = false;\n' 'window.jspython = pyodide.pyimport("open");\n' 'window.jsbytes = new Uint8Array([1, 2, 3]);\n' 'window.jsfloats = new Float32Array([1, 2, 3]);\n' 'window.jsobject = new XMLHttpRequest();\n' ) assert selenium.run( 'from js import jsstring\n' 'jsstring == "碘化物"') assert selenium.run( 'from js import jsnumber0\n' 'jsnumber0 == 42') assert selenium.run( 'from js import jsnumber1\n' 'jsnumber1 == 42.5') assert selenium.run( 'from js import jsundefined\n' 'jsundefined is None') assert selenium.run( 'from js import jstrue\n' 'jstrue is True') assert selenium.run( 'from js import jsfalse\n' 'jsfalse is False') assert selenium.run( 'from js import jspython\n' 'jspython is open') assert selenium.run( 'from js import jsbytes\n' '(jsbytes.tolist() == [1, 2, 3]) ' 'and (jsbytes.tobytes() == b"\x01\x02\x03")') assert selenium.run( 'from js import jsfloats\n' 'import struct\n' 'expected = struct.pack("fff", 1, 2, 3)\n' '(jsfloats.tolist() == [1, 2, 3]) ' 'and (jsfloats.tobytes() == expected)') assert selenium.run( 'from js import jsobject\n' 'str(jsobject) == "[object XMLHttpRequest]"') def test_typed_arrays(selenium): for wasm_heap in (False, True): for (jstype, pytype) in ( ('Int8Array', 'b'), ('Uint8Array', 'B'), ('Uint8ClampedArray', 'B'), ('Int16Array', 'h'), ('Uint16Array', 'H'), ('Int32Array', 'i'), ('Uint32Array', 'I'), ('Float32Array', 'f'), ('Float64Array', 'd')): print(wasm_heap, jstype, pytype) if not wasm_heap: selenium.run_js( f'window.array = new {jstype}([1, 2, 3, 4]);\n') else: selenium.run_js( 'var buffer = pyodide._malloc(' f'4 * {jstype}.BYTES_PER_ELEMENT);\n' f'window.array = new {jstype}(' 'pyodide.HEAPU8.buffer, buffer, 4);\n' 'window.array[0] = 1;\n' 'window.array[1] = 2;\n' 'window.array[2] = 3;\n' 'window.array[3] = 4;\n') assert selenium.run( 'from js import array\n' 'import struct\n' f'expected = struct.pack("{pytype*4}", 1, 2, 3, 4)\n' 'print(array.format, array.tolist(), array.tobytes())\n' f'array.format == "{pytype}" ' 'and array.tolist() == [1, 2, 3, 4] ' 'and array.tobytes() == expected ' f'and array.obj._has_bytes() is {not wasm_heap}') def test_import_js(selenium): result = selenium.run( "from js import window\nwindow.title = 'Foo'\nwindow.title") assert result == 'Foo' def test_pyproxy(selenium): selenium.run( "class Foo:\n" " bar = 42\n" " def get_value(self, value):\n" " return value * 64\n" "f = Foo()\n" ) assert selenium.run_js("return pyodide.pyimport('f').get_value(2)") == 128 assert selenium.run_js("return pyodide.pyimport('f').bar") == 42 assert selenium.run_js("return ('bar' in pyodide.pyimport('f'))") selenium.run_js("f = pyodide.pyimport('f'); f.baz = 32") assert selenium.run("f.baz") == 32 assert set(selenium.run_js( "return Object.getOwnPropertyNames(pyodide.pyimport('f'))")) == set( ['__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__init_subclass__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'bar', 'baz', 'get_value', 'toString', 'prototype', 'arguments', 'caller']) assert selenium.run("hasattr(f, 'baz')") selenium.run_js("delete pyodide.pyimport('f').baz") assert not selenium.run("hasattr(f, 'baz')") assert selenium.run_js( "return pyodide.pyimport('f').toString()").startswith('