Convert your Streamlit application into a desktop app with Stlite runtime, a Pyodide-based Wasm-port of Streamlit.
- Create the following
package.json
file to start a new NPM project. Edit thename
field.{ "name": "xxx", "version": "0.1.0", "main": "./build/electron/main.js", "scripts": { "dump": "dump-stlite-desktop-artifacts", "serve": "cross-env NODE_ENV=production electron .", "app:dir": "electron-builder --dir", "app:dist": "electron-builder", "postinstall": "electron-builder install-app-deps" }, "build": { "files": ["build/**/*"], "directories": { "buildResources": "assets" } }, "devDependencies": { "@stlite/desktop": "^0.69.2", "cross-env": "^7.0.3", "electron": "31.0.0", "electron-builder": "^25.1.7" }, "stlite": { "desktop": { "files": ["app.py"], "entrypoint": "app.py" } } }
- Run
npm install
oryarn install
. - Create
app.py
and write your Streamlit app code in it. The file nameapp.py
is specified both in thestlite.desktop.files
field in thepackage.json
and thestlite.desktop.entrypoint
field. If you want to use a different file name, change the file name in both fields.stlite.desktop.files
specifies the files and directories to be copied to the bundled desktop app.stlite.desktop.entrypoint
specifies the entry point of the Streamlit app.
- You can add more files and directories, such as
pages/*.py
for multi-page apps, any data files, and so on, by adding them to thestlite.desktop.files
field in thepackage.json
.{ // ...other fields... "stlite": { "desktop": { // ...other fields... "files": ["app.py", "pages/*.py", "assets"] } } }
- You can specify the packages to install in the desktop app by adding
stlite.desktop.dependencies
and/orstlite.desktop.requirementsTxtFiles
fields in thepackage.json
.stlite.desktop.dependencies
is an array of package names to install.{ // ...other fields... "stlite": { "desktop": { // ...other fields... "dependencies": ["numpy", "pandas"] } } }
stlite.desktop.requirementsTxtFiles
is an array of paths torequirements.txt
files to install the packages listed in the files.{ // ...other fields... "stlite": { "desktop": { // ...other fields... "requirementsTxtFiles": ["requirements.txt"] } } }
- Run
npm run dump
oryarn dump
.- This
dump
command creates./build
directory that contains the copied Streamlit app files, dumped installed packages, Pyodide runtime, Electron app files, etc.
- This
- Run
npm run serve
oryarn serve
for preview.- This command is just a wrapper of
electron
command as you can see at the"scripts"
field in thepackage.json
. It launches Electron and starts the app with./build/electron/main.js
, which is specified at the"main"
field in thepackage.json
.
- This command is just a wrapper of
- Run
npm run app:dist
oryarn app:dist
for packaging.- This command bundles the
./build
directory created in the step above into application files (.app
,.exe
,.dmg
etc.) in the./dist
directory. To customize the built app, e.g. setting the icon, follow theelectron-builder
instructions.
- This command bundles the
See the ./samples directory for sample projects.
To make your app secure, be sure to use the latest version of Electron. This is announced as one of the security best practices in the Electron document too.
The dump
command downloads some Pyodide resources such as the prebuilt package wheel files from the JsDelivr CDN by default.
If you want to use a different Pyodide source, for example when accessing JsDelivr (cdn.jsdelivr.net
) is restricted in your environment,
you can specify a URL or a path to the Pyodide source by setting the --pyodide-source
option of the dump
command.
For example, if you downloaded a Pyodide package from the Pyodide releases and saved it in /path/to/pyodide/
, you can specify the URL to the Pyodide package like below.
npm run dump -- --pyodide-source /path/to/pyodide/
yarn dump --pyodide-source /path/to/pyodide/
If you want to hide the toolbar, hamburger menu, and footer, add the following to your package.json
file and run the dump
command again. By adding the stlite.desktop.embed
field, the dumped Streamlit app will work in the embed mode which hides the toolbar, hamburger menu, and footer.
{
// ...other fields...
"stlite": {
"desktop": {
"embed": true
}
}
}
Stlite runs your Python code on Pyodide, a CPython runtime compiled to Wasm, and Pyodide's backend, Emscripten, provides a virtual file system.
When Stlite runs your app, it mounts the source files onto the virtual file system, and what your Python code can access (e.g. open("/path/to/something")
) is files and directories on the virtual file system.
The default file system (MEMFS
) is ephemeral, so the files saved in the directories are lost when the app is restarted. If you want to persist the files across the app restarts, you can use the IndexedDB-based file system (IDBFS
) or mount directories on the host OS file system to directories on the virtual file system.
You can mount the IndexedDB-based file system (IDBFS
) to directories on the virtual file system that your Python code can access, e.g. open("/path/to/file")
.
You can specify the mount points via the stlite.desktop.idbfsMountpoints
field in your package.json
like below.
Note that you have to run the dump
command again to apply the change.
The mounted file system is backed by IndexedDB and its data is stored in the browser's IndexedDB, so the files saved in the directories are persistent across the app restarts.
In the example below, the IndexedDB-based file system is mounted to the /mnt
directory on the virtual file system, so that the files saved in the directory are persistent.
{
// ...other fields...
"stlite": {
"desktop": {
"idbfsMountpoints": ["/mnt"]
}
}
}
You can mount directories on the host OS file system to directories on the virtual file system.
To do this, you have to enable the Node.js worker mode (see the next section for details) and specify the mount points via the stlite.desktop.nodefsMountpoints
field in your package.json
like below.
The nodefsMountpoints
field is an object that maps the virtual file system paths to the host OS paths.
In the example below, "."
on the host OS file system is mounted to the /mnt
directory on the virtual file system, so your app can access the files in "."
on the host OS by accessing the files in /mnt
on the virtual file system.
{
// ...other fields...
"stlite": {
"desktop": {
"nodeJsWorker": true,
"nodefsMountpoints": {
"/mnt": "."
}
}
}
}
You can use the placeholders such as {{home}}
, {{userData}}
, {{temp}}
, etc. in the host OS paths to specify the paths dynamically. Check the Electron's app.getPath
documentation for the available path names because the placeholders are resolved to the paths returned by app.getPath
.
{
// ...other fields...
"stlite": {
"desktop": {
"nodeJsWorker": true,
"nodefsMountpoints": {
"/foo": "{{home}}/foo" // e.g. The host OS path is resolved to "/home/user/foo" on Linux
}
}
}
}
@stlite/desktop
runs your app on Electron as a desktop app.
Electron apps have two processes: the main process which is a Node.js process running in the background, and the renderer process which is a Chromium (browser) process running the app's UI.
By default, Stlite executes your Python code on Pyodide running in a Web Worker dispatched by the renderer process, and the renderer process is a browser process so it's sandboxed from the host OS.
When you set the stlite.desktop.nodeJsWorker
field in your package.json
to true
, Stlite dispatches the worker as a NodeJS worker that runs in the main process, which is not sandboxed, so you can mount the host OS file system to the virtual file system as described in the previous section.
{
// ...other fields...
"stlite": {
"desktop": {
"nodeJsWorker": true
}
}
}
- Navigation to external resources like
st.markdown("[link](https://streamlit.io/)")
does not work for security. See #445 and let us know if you have use cases where you have to use such external links.