https://github.com/metapages/metaframe-py
Python metaframe via pyiodide
https://github.com/metapages/metaframe-py
metaframe pyodide python
Last synced: 4 months ago
JSON representation
Python metaframe via pyiodide
- Host: GitHub
- URL: https://github.com/metapages/metaframe-py
- Owner: metapages
- Created: 2023-11-09T19:10:41.000Z (over 1 year ago)
- Default Branch: main
- Last Pushed: 2023-11-11T20:47:41.000Z (over 1 year ago)
- Last Synced: 2025-01-02T14:25:22.632Z (5 months ago)
- Topics: metaframe, pyodide, python
- Language: HTML
- Homepage: https://py.mtfm.io
- Size: 1.23 MB
- Stars: 0
- Watchers: 1
- Forks: 0
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
Awesome Lists containing this project
README
# Python [metaframe](https://docs.metapage.io/docs) (via pyiodide)
```mermaid
graph LR
classDef mpColor fill:#87d0ff,stroke:#87d0ff,stroke-width:2px;
subgraph metapage
direction LR
left1(upstream metaframe) -->|inputs| M(python code):::mpColor
M --> |outputs| right1(downstream metaframe)
end```
Examples:
- [create matplotlib plots](https://app.metapages.org/#?definition=%7B%22meta%22%3A%7B%22layouts%22%3A%7B%22react-grid-layout%22%3A%7B%22docs%22%3A%22https%3A%2F%2Fwww.npmjs.com%2Fpackage%2Freact-grid-layout%22%2C%22layout%22%3A%5B%7B%22h%22%3A3%2C%22i%22%3A%22image1%22%2C%22moved%22%3Afalse%2C%22static%22%3Afalse%2C%22w%22%3A12%2C%22x%22%3A0%2C%22y%22%3A2%7D%2C%7B%22h%22%3A5%2C%22i%22%3A%22image2%22%2C%22moved%22%3Afalse%2C%22static%22%3Afalse%2C%22w%22%3A12%2C%22x%22%3A0%2C%22y%22%3A5%7D%2C%7B%22h%22%3A2%2C%22i%22%3A%22pyiodide-dev%22%2C%22moved%22%3Afalse%2C%22static%22%3Afalse%2C%22w%22%3A12%2C%22x%22%3A0%2C%22y%22%3A0%7D%5D%2C%22props%22%3A%7B%22cols%22%3A12%2C%22containerPadding%22%3A%5B5%2C5%5D%2C%22margin%22%3A%5B10%2C20%5D%2C%22rowHeight%22%3A100%7D%7D%7D%2C%22name%22%3A%22A%20new%20kind%20of%20publishing%20-%20self-contained%20URLS%22%7D%2C%22metaframes%22%3A%7B%22image1%22%3A%7B%22inputs%22%3A%5B%7B%22metaframe%22%3A%22pyiodide-dev%22%2C%22source%22%3A%22image1%22%2C%22target%22%3A%22image1%22%7D%5D%2C%22url%22%3A%22https%3A%2F%2Fjs.mtfm.io%2F%23%3Fedit%3D1%26js%3DY29uc3QlMjByb290JTIwJTNEJTIwZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJTIycm9vdCUyMiklMEElMEF2YXIlMjBvbklucHV0cyUyMCUzRCUyMGlucHV0cyUyMCUzRCUzRSUyMCU3QiUwQSUyMCUyMCUyMCUyMGNvbnN0JTIwayUyMCUzRCUyME9iamVjdC5rZXlzKGlucHV0cyklNUIwJTVEJTNCJTBBJTIwJTIwJTIwJTIwaWYlMjAoIShrJTIwJTI2JTI2JTIwaW5wdXRzJTVCayU1RCkpJTIwJTdCJTBBJTIwJTIwJTIwJTIwJTIwJTIwJTIwJTIwcmV0dXJuJTNCJTBBJTIwJTIwJTIwJTIwJTdEJTBBJTIwJTIwJTIwJTIwY29uc3QlMjB2JTIwJTNEJTIwaW5wdXRzJTVCayU1RCUwQSUyMCUyMCUyMCUyMHZhciUyMGltYWdlJTIwJTNEJTIwZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnaW1nJyklM0IlMEElMDlpbWFnZS5zcmMlMjAlM0QlMjB0eXBlb2YodiklMjAlM0QlM0QlM0QlMjAlMjJzdHJpbmclMjIlMjAlMjYlMjYlMjB2LnN0YXJ0c1dpdGgoJTIyZGF0YSUzQWltYWdlJTIyKSUyMCUzRiUyMHYlMjAlM0ElMjBVUkwuY3JlYXRlT2JqZWN0VVJMKHYpJTNCJTBBJTIwJTIwJTIwJTIwd2hpbGUlMjAocm9vdC5maXJzdENoaWxkKSUyMCU3QiUwQSUyMCUyMCUyMCUyMCUyMCUyMCUyMCUyMHJvb3QucmVtb3ZlQ2hpbGQocm9vdC5sYXN0Q2hpbGQpJTNCJTBBJTIwJTIwJTIwJTIwJTdEJTBBJTIwJTIwJTIwJTIwcm9vdC5hcHBlbmRDaGlsZChpbWFnZSklM0IlMEElN0QlMEElMEFvbklucHV0cyhtZXRhZnJhbWUuZ2V0SW5wdXRzKCkpJTBBJTBBY29uc3QlMjBkaXNwb3NlJTIwJTNEJTIwbWV0YWZyYW1lLm9uSW5wdXRzKG9uSW5wdXRzKSUwQXdpbmRvdy5zY3JpcHRVbmxvYWQlMjAlM0QlMjAoKSUyMCUzRCUzRSUyMCU3QiUwQSUwOWRpc3Bvc2UoKSUwQSU3RA%3D%3D%26modules%3DJTVCJTVE%22%7D%2C%22image2%22%3A%7B%22inputs%22%3A%5B%7B%22metaframe%22%3A%22pyiodide-dev%22%2C%22source%22%3A%22image2%22%2C%22target%22%3A%22image2%22%7D%5D%2C%22url%22%3A%22https%3A%2F%2Fjs.mtfm.io%2F%23%3Fedit%3D1%26js%3DY29uc3QlMjByb290JTIwJTNEJTIwZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJTIycm9vdCUyMiklMEElMEF2YXIlMjBvbklucHV0cyUyMCUzRCUyMGlucHV0cyUyMCUzRCUzRSUyMCU3QiUwQSUyMCUyMCUyMCUyMGNvbnN0JTIwayUyMCUzRCUyME9iamVjdC5rZXlzKGlucHV0cyklNUIwJTVEJTNCJTBBJTIwJTIwJTIwJTIwaWYlMjAoIShrJTIwJTI2JTI2JTIwaW5wdXRzJTVCayU1RCkpJTIwJTdCJTBBJTIwJTIwJTIwJTIwJTIwJTIwJTIwJTIwcmV0dXJuJTNCJTBBJTIwJTIwJTIwJTIwJTdEJTBBJTIwJTIwJTIwJTIwY29uc3QlMjB2JTIwJTNEJTIwaW5wdXRzJTVCayU1RCUwQSUyMCUyMCUyMCUyMHZhciUyMGltYWdlJTIwJTNEJTIwZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnaW1nJyklM0IlMEElMDlpbWFnZS5zcmMlMjAlM0QlMjB0eXBlb2YodiklMjAlM0QlM0QlM0QlMjAlMjJzdHJpbmclMjIlMjAlMjYlMjYlMjB2LnN0YXJ0c1dpdGgoJTIyZGF0YSUzQWltYWdlJTIyKSUyMCUzRiUyMHYlMjAlM0ElMjBVUkwuY3JlYXRlT2JqZWN0VVJMKHYpJTNCJTBBJTIwJTIwJTIwJTIwd2hpbGUlMjAocm9vdC5maXJzdENoaWxkKSUyMCU3QiUwQSUyMCUyMCUyMCUyMCUyMCUyMCUyMCUyMHJvb3QucmVtb3ZlQ2hpbGQocm9vdC5sYXN0Q2hpbGQpJTNCJTBBJTIwJTIwJTIwJTIwJTdEJTBBJTIwJTIwJTIwJTIwcm9vdC5hcHBlbmRDaGlsZChpbWFnZSklM0IlMEElN0QlMEElMEFvbklucHV0cyhtZXRhZnJhbWUuZ2V0SW5wdXRzKCkpJTBBJTBBY29uc3QlMjBkaXNwb3NlJTIwJTNEJTIwbWV0YWZyYW1lLm9uSW5wdXRzKG9uSW5wdXRzKSUwQXdpbmRvdy5zY3JpcHRVbmxvYWQlMjAlM0QlMjAoKSUyMCUzRCUzRSUyMCU3QiUwQSUwOWRpc3Bvc2UoKSUwQSU3RA%3D%3D%26modules%3DJTVCJTVE%22%7D%2C%22pyiodide-dev%22%3A%7B%22inputs%22%3A%5B%7B%22metaframe%22%3A%22input%22%2C%22source%22%3A%22text%22%2C%22target%22%3A%22text%22%7D%2C%7B%22metaframe%22%3A%22generate-inputs%22%2C%22source%22%3A%22*%22%2C%22target%22%3A%22*%22%7D%5D%2C%22url%22%3A%22https%3A%2F%2Fpy.mtfm.io%2F%23%3Fedit%3D1%26options%3DJTdCJTIyciUyMiUzQXRydWUlN0Q%3D%26py%3DaW1wb3J0JTIwcHlvZGlkZSUwQWltcG9ydCUyMG1pY3JvcGlwJTBBZnJvbSUyMGpzJTIwaW1wb3J0JTIwbWV0YWZyYW1lJTJDJTIwbG9nJTBBJTBBYXdhaXQlMjBtaWNyb3BpcC5pbnN0YWxsKCUyMm1hdHBsb3RsaWIlMjIpJTBBYXdhaXQlMjBtaWNyb3BpcC5pbnN0YWxsKCUyMm51bXB5JTIyKSUwQSUwQWltcG9ydCUyMG1hdHBsb3RsaWIucHlwbG90JTIwYXMlMjBwbHQlMEFpbXBvcnQlMjBpbyUyQyUyMGJhc2U2NCUwQWltcG9ydCUyMG51bXB5JTIwYXMlMjBucCUwQSUwQWxvZyglMjJQeXRob24lMjBzdGFydGVkJTJDJTIwY29tcHV0aW5nJTIwcGxvdHMuLi4lMjIpJTBBJTBBbnAucmFuZG9tLnNlZWQoMTk2ODA4MDEpJTIwJTIwJTIzJTIwc2VlZCUyMHRoZSUyMHJhbmRvbSUyMG51bWJlciUyMGdlbmVyYXRvci4lMEFkYXRhJTIwJTNEJTIwJTdCJ2EnJTNBJTIwbnAuYXJhbmdlKDUwKSUyQyUwQSUyMCUyMCUyMCUyMCUyMCUyMCUyMCUyMCdjJyUzQSUyMG5wLnJhbmRvbS5yYW5kaW50KDAlMkMlMjA1MCUyQyUyMDUwKSUyQyUwQSUyMCUyMCUyMCUyMCUyMCUyMCUyMCUyMCdkJyUzQSUyMG5wLnJhbmRvbS5yYW5kbig1MCklN0QlMEFkYXRhJTVCJ2InJTVEJTIwJTNEJTIwZGF0YSU1QidhJyU1RCUyMCUyQiUyMDEwJTIwKiUyMG5wLnJhbmRvbS5yYW5kbig1MCklMEFkYXRhJTVCJ2QnJTVEJTIwJTNEJTIwbnAuYWJzKGRhdGElNUInZCclNUQpJTIwKiUyMDEwMCUwQSUwQWZpZyUyQyUyMGF4JTIwJTNEJTIwcGx0LnN1YnBsb3RzKGZpZ3NpemUlM0QoNSUyQyUyMDIuNyklMkMlMjBsYXlvdXQlM0QnY29uc3RyYWluZWQnKSUwQWF4LnNjYXR0ZXIoJ2EnJTJDJTIwJ2InJTJDJTIwYyUzRCdjJyUyQyUyMHMlM0QnZCclMkMlMjBkYXRhJTNEZGF0YSklMEFheC5zZXRfeGxhYmVsKCdlbnRyeSUyMGEnKSUwQWF4LnNldF95bGFiZWwoJ2VudHJ5JTIwYicpJTBBJTBBYnVmJTIwJTNEJTIwaW8uQnl0ZXNJTygpJTBBZmlnLnNhdmVmaWcoYnVmJTJDJTIwZm9ybWF0JTNEJ3BuZycpJTBBYnVmLnNlZWsoMCklMEFpbWdfc3RyJTIwJTNEJTIwJ2RhdGElM0FpbWFnZSUyRnBuZyUzQmJhc2U2NCUyQyclMjAlMkIlMjBiYXNlNjQuYjY0ZW5jb2RlKGJ1Zi5yZWFkKCkpLmRlY29kZSgnVVRGLTgnKSUwQWxvZyglMjJTZW50JTIwaW1hZ2UxISUyMiklMEFtZXRhZnJhbWUuc2V0T3V0cHV0KCUyMmltYWdlMSUyMiUyQyUyMGltZ19zdHIpJTBBJTBBJTBBZmlnJTJDJTIwYXglMjAlM0QlMjBwbHQuc3VicGxvdHMoKSUwQWF4LnBsb3QoJTVCMSUyQzMlMkMyJTVEKSUwQWJ1ZiUyMCUzRCUyMGlvLkJ5dGVzSU8oKSUwQWZpZy5zYXZlZmlnKGJ1ZiUyQyUyMGZvcm1hdCUzRCdwbmcnKSUwQWJ1Zi5zZWVrKDApJTBBaW1nX3N0ciUyMCUzRCUyMCdkYXRhJTNBaW1hZ2UlMkZwbmclM0JiYXNlNjQlMkMnJTIwJTJCJTIwYmFzZTY0LmI2NGVuY29kZShidWYucmVhZCgpKS5kZWNvZGUoJ1VURi04JyklMEFsb2coJTIyU2VudCUyMGltYWdlMiElMjIpJTBBbWV0YWZyYW1lLnNldE91dHB1dCglMjJpbWFnZTIlMjIlMkMlMjBpbWdfc3RyKQ%3D%3D%22%7D%7D%2C%22plugins%22%3A%5B%22https%3A%2F%2Fmermaid.mtfm.io%2F%23%3Fhm%3Ddisabled%22%2C%22https%3A%2F%2Feditor.mtfm.io%2F%23%3Fhm%3Ddisabled%26options%3DJTdCJTIybW9kZSUyMiUzQSUyMmpzb24lMjIlMkMlMjJzYXZlbG9hZGluaGFzaCUyMiUzQXRydWUlMkMlMjJ0aGVtZSUyMiUzQSUyMnZzLWRhcmslMjIlN0Q%3D%22%5D%2C%22version%22%3A%220.3%22%7D)
- [inputs/outputs](https://app.metapage.io/dion/py.mtfm.io-test/view)- Run python code entire in the browser, and connect to other metaframes
- Edit python code directly in the frame
- All code and configuration is stored in the URL
- shareable and durable## Configuration + Inputs/Outputs
There are two (metaframe) modes (set in ⚙️ Options):
1. `✅ Re-run python code on new inputs`: Anytime new metaframe inputs arrive, restart the script. Any files in `/outputs` are copied and send as metaframe outputs. If the file suffix is recognized (e.g. `.js`, `.json`, `.py`) then it will be sent as a string (or directly as JSON if ending in `.json`), otherwise it will be send as an ArrayBuffer.
- Inputs **are** copied as files
2. `❌ Re-run python code on new inputs`: Run the script once, the script is responsible for listening to inputs by calling the metaframe object (see examples below).
- Inputs are **not** copied as files
The use case of copying inputs and outputs as files but **not* re-running seems counterintuitive to me, let me know if you have this use case: https://github.com/metapages/metaframe-py/issues## Useful code snippets
### Get the (current) metaframe inputs
```python
from js import metaframeinputs = metaframe.getInputs().as_object_map()
for key in inputs:
print(key)
print(inputs[key])
# Do something with the value
```### Listen to new metaframe inputs
```python
from pyodide.ffi import create_proxy
from js import metaframedef onInputs(*args):
print('got inputs in python:')
inputs = args[0].as_object_map()
for key in inputs:
print(key)onInputsProxy = create_proxy(onInputs)
disposer = metaframe.onInputs(onInputsProxy)
# Call the disposer via the scriptUnload described next below```
### Unload/cleanup/dispose
When iterating/editing, the python script is repeatedly run in the `pyiodide` object. This can cause problems if proxied `js <-> python` objects are not properly disposed.
Before every new python script execution, (in javascript) if a `window.scriptUnload` script exists, it is called. This is where you put your cleanup:
```python
from pyodide.ffi import create_proxy
from js import window# Put this at the end of your script:
def scriptUnload(*args):
print('unloading from python')
# Call your cleanup code here
proxy_scriptUnload = create_proxy(scriptUnload)
window.scriptUnload = proxy_scriptUnload
```### Logging to the main div
Some convenience functions:
```py
# log is the same as logStderr
from js import log, logStdout, logStderr# These functions only take a single string, no fancy formatting or multiple arguments
log("some string")
logStdout("some string")
logStderr("some error")
```## Advanced
- Testing, input/outputs, capabilities, development metapage:
- https://app.metapage.io/dion/py.mtfm.io-test/view
- Example metapage running python with inputs and outputs and plotting:
- https://app.metapage.io/dion/py.mtfm.io-example-01