Skip to content

Latest commit

 

History

History
218 lines (180 loc) · 9.92 KB

README.md

File metadata and controls

218 lines (180 loc) · 9.92 KB

@stlite/desktop

Convert your Streamlit application into a desktop app with Stlite runtime, a Pyodide-based Wasm-port of Streamlit.

How to create a Streamlit desktop app

  1. Create the following package.json file to start a new NPM project. Edit the name 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"
        }
      }
    }
  2. Run npm install or yarn install.
  3. Create app.py and write your Streamlit app code in it. The file name app.py is specified both in the stlite.desktop.files field in the package.json and the stlite.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.
  4. 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 the stlite.desktop.files field in the package.json.
    {
      // ...other fields...
      "stlite": {
        "desktop": {
          // ...other fields...
          "files": ["app.py", "pages/*.py", "assets"]
        }
      }
    }
  5. You can specify the packages to install in the desktop app by adding stlite.desktop.dependencies and/or stlite.desktop.requirementsTxtFiles fields in the package.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 to requirements.txt files to install the packages listed in the files.
      {
        // ...other fields...
        "stlite": {
          "desktop": {
            // ...other fields...
            "requirementsTxtFiles": ["requirements.txt"]
          }
        }
      }
  6. Run npm run dump or yarn dump.
    • This dump command creates ./build directory that contains the copied Streamlit app files, dumped installed packages, Pyodide runtime, Electron app files, etc.
  7. Run npm run serve or yarn serve for preview.
    • This command is just a wrapper of electron command as you can see at the "scripts" field in the package.json. It launches Electron and starts the app with ./build/electron/main.js, which is specified at the "main" field in the package.json.
  8. Run npm run app:dist or yarn 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 the electron-builder instructions.

See the ./samples directory for sample projects.

Use the latest version of Electron

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.

Use a custom Pyodide source

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/

Configure the app

Hide the toolbar, hamburger menu, and the footer

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
    }
  }
}

File system

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.

File persistence with IndexedDB backend

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"]
    }
  }
}

Local file access

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
      }
    }
  }
}

NodeJS worker mode

@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
    }
  }
}

Limitations

  • 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.