2018-06-21 15:19:34 +00:00
|
|
|
# Using Pyodide from Javascript
|
|
|
|
|
|
|
|
This document describes using Pyodide directly from Javascript. For information
|
|
|
|
about using Pyodide from Iodide, see [Using Pyodide from
|
|
|
|
Iodide](using_pyodide_from_iodide.md).
|
|
|
|
|
|
|
|
## Startup
|
|
|
|
|
2020-05-20 16:58:43 +00:00
|
|
|
To include Pyodide in your project you can use the following CDN URL,
|
2018-06-21 15:19:34 +00:00
|
|
|
|
2020-05-20 19:11:50 +00:00
|
|
|
https://pyodide-cdn2.iodide.io/v0.15.0/full/pyodide.js
|
2019-05-31 20:02:51 +00:00
|
|
|
|
2020-05-20 16:58:43 +00:00
|
|
|
You can also download a release from
|
|
|
|
[Github releases](https://github.com/iodide-project/pyodide/releases)
|
|
|
|
(or build it yourself), include its contents in your distribution, and import
|
|
|
|
the `pyodide.js` file there from a `<script>` tag. See the following section on
|
|
|
|
[serving pyodide files](./#serving-pyodide-files) for more details.
|
2019-05-31 20:02:51 +00:00
|
|
|
|
2020-05-20 16:58:43 +00:00
|
|
|
The `pyodide.js` file has a single `Promise` object which bootstraps the Python
|
|
|
|
environment: `languagePluginLoader`. Since this must happen asynchronously, it
|
|
|
|
is a `Promise`, which you must call `then` on to complete initialization. When
|
|
|
|
the promise resolves, pyodide will have installed a namespace in global scope:
|
2018-06-21 15:19:34 +00:00
|
|
|
`pyodide`.
|
|
|
|
|
|
|
|
```javascript
|
2018-11-08 13:25:33 +00:00
|
|
|
languagePluginLoader.then(() => {
|
2018-06-21 15:19:34 +00:00
|
|
|
// pyodide is now ready to use...
|
|
|
|
console.log(pyodide.runPython('import sys\nsys.version'));
|
|
|
|
});
|
|
|
|
```
|
|
|
|
|
|
|
|
## Running Python code
|
|
|
|
|
|
|
|
Python code is run using the `pyodide.runPython` function. It takes as input a
|
|
|
|
string of Python code. If the code ends in an expression, it returns the result
|
|
|
|
of the expression, converted to Javascript objects (See [type
|
|
|
|
conversions](type_conversions.md)).
|
|
|
|
|
|
|
|
```javascript
|
2020-07-13 19:46:20 +00:00
|
|
|
pyodide.runPython(`
|
|
|
|
import sys
|
|
|
|
sys.version
|
|
|
|
`);
|
2018-06-21 15:19:34 +00:00
|
|
|
```
|
|
|
|
|
2020-05-20 16:58:43 +00:00
|
|
|
## Complete example
|
|
|
|
|
|
|
|
Create and save a test `index.html` page with the following contents:
|
|
|
|
```html
|
|
|
|
<!DOCTYPE html>
|
|
|
|
<html>
|
2020-07-13 19:46:20 +00:00
|
|
|
<head>
|
|
|
|
<script type="text/javascript">
|
|
|
|
// set the pyodide files URL (packages.json, pyodide.asm.data etc)
|
|
|
|
window.languagePluginUrl = 'https://pyodide-cdn2.iodide.io/v0.15.0/full/';
|
|
|
|
</script>
|
|
|
|
<script src="https://pyodide-cdn2.iodide.io/v0.15.0/full/pyodide.js"></script>
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
Pyodide test page <br>
|
|
|
|
Open your browser console to see pyodide output
|
2020-05-20 16:58:43 +00:00
|
|
|
<script type="text/javascript">
|
2020-07-13 19:46:20 +00:00
|
|
|
languagePluginLoader.then(function () {
|
|
|
|
console.log(pyodide.runPython(`
|
|
|
|
import sys
|
|
|
|
sys.version
|
|
|
|
`));
|
|
|
|
console.log(pyodide.runPython('print(1 + 2)'));
|
|
|
|
});
|
2020-05-20 16:58:43 +00:00
|
|
|
</script>
|
2020-07-13 19:46:20 +00:00
|
|
|
</body>
|
|
|
|
</html>
|
2020-05-20 16:58:43 +00:00
|
|
|
```
|
|
|
|
|
2018-06-21 15:19:34 +00:00
|
|
|
## Loading packages
|
|
|
|
|
|
|
|
Only the Python standard library and `six` are available after importing
|
|
|
|
Pyodide. To use other libraries, you'll need to load their package using
|
|
|
|
`pyodide.loadPackage`. This downloads the file data over the network (as a
|
|
|
|
`.data` and `.js` index file) and installs the files in the virtual filesystem.
|
|
|
|
|
2018-08-29 08:40:10 +00:00
|
|
|
Packages can be loaded by name, for those included in the official pyodide
|
|
|
|
repository (e.g. `pyodide.loadPackage('numpy')`). It is also possible to load
|
|
|
|
packages from custom URLs (e.g.
|
|
|
|
`pyodide.loadPackage('https://foo/bar/numpy.js')`), in which case the URL must
|
|
|
|
end with `<package-name>.js`.
|
|
|
|
|
|
|
|
When you request a package from the official repository, all of that package's
|
|
|
|
dependencies are also loaded. Dependency resolution is not yet implemented
|
|
|
|
when loading packages from custom URLs.
|
2018-06-21 15:19:34 +00:00
|
|
|
|
2018-09-05 09:28:20 +00:00
|
|
|
Multiple packages can also be loaded in a single call,
|
|
|
|
```js
|
|
|
|
pyodide.loadPackage(['cycler', 'pytz'])
|
|
|
|
```
|
|
|
|
|
2018-06-21 15:19:34 +00:00
|
|
|
`pyodide.loadPackage` returns a `Promise`.
|
|
|
|
|
|
|
|
```javascript
|
|
|
|
pyodide.loadPackage('matplotlib').then(() => {
|
2019-03-29 19:03:51 +00:00
|
|
|
// matplotlib is now available
|
2018-06-21 15:19:34 +00:00
|
|
|
});
|
|
|
|
```
|
|
|
|
|
2020-05-20 16:58:43 +00:00
|
|
|
## Serving pyodide files
|
2020-03-25 11:29:38 +00:00
|
|
|
|
2020-05-20 16:58:43 +00:00
|
|
|
If you built your pyodide distribution or downloaded the release tarball
|
|
|
|
you need to serve pyodide files with a appropriate headers.
|
2020-03-25 11:29:38 +00:00
|
|
|
|
|
|
|
Because browsers require WebAssembly files to have mimetype of
|
|
|
|
`application/wasm` we're unable to serve our files using Python's built-in
|
|
|
|
`SimpleHTTPServer` module.
|
|
|
|
|
|
|
|
Let's wrap Python's Simple HTTP Server and provide the appropiate mimetype for
|
|
|
|
WebAssembly files into a `pyodide_server.py` file (in the `pyodide_local`
|
|
|
|
directory):
|
|
|
|
```python
|
|
|
|
import sys
|
|
|
|
import socketserver
|
|
|
|
from http.server import SimpleHTTPRequestHandler
|
|
|
|
|
|
|
|
|
|
|
|
class Handler(SimpleHTTPRequestHandler):
|
|
|
|
|
|
|
|
def end_headers(self):
|
|
|
|
# Enable Cross-Origin Resource Sharing (CORS)
|
|
|
|
self.send_header('Access-Control-Allow-Origin', '*')
|
|
|
|
super().end_headers()
|
|
|
|
|
|
|
|
|
|
|
|
if sys.version_info < (3, 7, 5):
|
|
|
|
# Fix for WASM MIME type for older Python versions
|
|
|
|
Handler.extensions_map['.wasm'] = 'application/wasm'
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
port = 8000
|
|
|
|
with socketserver.TCPServer(("", port), Handler) as httpd:
|
|
|
|
print("Serving at: http://127.0.0.1:{}".format(port))
|
|
|
|
httpd.serve_forever()
|
|
|
|
```
|
|
|
|
|
|
|
|
Let's test it out.
|
|
|
|
In your favourite shell, let's start our WebAssembly aware web server:
|
|
|
|
```bash
|
|
|
|
python pyodide_server.py
|
|
|
|
```
|
|
|
|
|
|
|
|
Point your WebAssembly aware browser to
|
|
|
|
[http://localhost:8000/index.html](http://localhost:8000/index.html) and open
|
|
|
|
your browser console to see the output from python via pyodide!
|