Skip to content

Latest commit

 

History

History
187 lines (119 loc) · 8.47 KB

BUNDLERS.md

File metadata and controls

187 lines (119 loc) · 8.47 KB

Deploying Local Vault WITH A Bundler

This project has non-ESM dependencies, which unfortunately cannot be bundled in with your other app code. Modern bundlers unfortunately don't out-of-the-box support configurations that can handle such a situation.

As such, this project provides plugins for Astro, Vite, and Webpack, to take care of the various steps needed to get these non-ESM dependencies into an otherwise bundled web app built by those tools.

Bundler Plugins

The plugins for Astro, Vite, and Webpack are included in the bundler-plugins/ directory. They should handle all necessary steps to load the dependencies.

Note: You should not need to manually copy any files, as the plugins access the dependencies (in node_modules) directly, to pull the files needed. But for reference, the files these plugins access are:

  • @lo-fi/local-vault/dist/bundlers/lv.mjs

    ESM library module that's suitable for bundling and importing into your web app.

    Note: this is not the same as @lo-fi/local-vault/dist/auto/lv.js, which is only intended for web application projects WITHOUT a bundler

  • @lo-fi/local-vault/dist/bundlers/base-adapter.mjs

  • storage adapters (despite naming similarities, these two sets of files are different, but related):

    • @lo-fi/local-vault/dist/bundlers/adapter.*.mjs

    • @byojs/storage/dist/adapter.*.mjs

  • @byojs/storage/dist/util.mjs

  • @byojs/storage/dist/worker.opfs.mjs

    ESM module loaded into a web worker, when using the opfs-worker adapter

  • @byojs/storage/dist/external/idb-keyval.js

  • @lo-fi/local-data-lock/dist/bundlers/ldl.mjs

  • @lo-fi/webauthn-local-client/dist/bundlers/walc.mjs

  • @lo-fi/webauthn-local-client/dist/bundlers/walc-external-bundle.js

    Non-ESM (plain global .js) bundle of dependencies that must be loaded separately from (and prior to) your app's bundle.

Astro Plugin (aka Integration)

If using Astro 4+, it's strongly suggested to import this library's Astro-plugin to manage the loading of its non-ESM dependencies. Add something like the following to your astro.config.mjs file:

import { defineConfig } from "astro/config";

import LV from "@lo-fi/local-vault/bundlers/astro";

export default defineConfig({
    // ..

    integrations: [ LV(), ],

    vite: {
        plugins: [
            // pulls in some necessary bits of LV's vite plugin
            LV.vite(),
        ],

        optimizeDeps: {
            esbuildOptions: {
                // LV's WALC dependency uses "top-level await", which is ES2022+
                target: "es2022",
            },
        },

        build: {
            // LV's WALC dependency uses "top-level await", which is ES2022+
            target: "es2022"
        },
    },

    // ..
});

This plugin works for the astro dev (dev-server), as well as astro build / astro preview modes. In all cases, it copies the @lo-fi/local-vault/dist/bundlers/walc-external-bundle.js file into the public/ directory of your project root, as well as the dist/ directory when running a build. It also injects an inline <script> element into the <head> of all generated pages, which dynamically loads the /walc-external-bundle.js script file (which has all the dependencies WALC needs).

Note: At present, this plugin is not configurable in any way (i.e., calling LV() above with no arguments). If something about its behavior is not compatible with your Astro project setup -- which can vary widely and be quite complex to predict or support by a basic plugin -- it's recommended you simply copy over the @lo-fi/local-vault/bundler-plugins/astro.mjs plugin and make necessary changes.

Vite Plugin

If using Vite 5+ directly, it's strongly suggested to import this library's Vite-plugin to manage the loading of its non-ESM dependencies. Add something like the following to your vite.config.js file:

import { defineConfig } from "vite";
import LV from "@lo-fi/local-vault/bundlers/vite";

export default defineConfig({
    // ..

    plugins: [ LV() ],

    optimizeDeps: {
        esbuildOptions: {
            // WALC (dependency) uses "top-level await", which is ES2022+
            target: "es2022",
        },
    },

    build: {
        // WALC (dependency) uses "top-level await", which is ES2022+
        target: "es2022"
    },

    // ..
});

This plugin works for the vite dev (dev-server), vite preview (also dev-server), and vite build modes. In all cases, it copies the @lo-fi/webauthn-local-client/dist/bundlers/walc-external-bundle.js file into the public/ directory of your project root. It also injects a <script src="/walc-external-bundle.js"></script> tag into the markup of the index.html file that Vite produces for your app.

Note: At present, this plugin is not configurable in any way (i.e., calling LV() above with no arguments). If something about its behavior is not compatible with your Vite project setup -- which can vary widely and be quite complex to predict or support by a basic plugin -- it's recommended you simply copy over the @lo-fi/local-vault/bundler-plugins/vite.mjs plugin and make necessary changes.

Top-level await

One of this library's transitivie dependencies (WebAuthn-Local-Client) uses "top-level await", a feature added to JS in ES2022. The current default target for Vite seems to be browsers older than this, so the above config explicitly sets the targets to "es2022".

You may experience issues where your tooling/configuration either ignores this setting, or otherwise breaks with it set. This may variously result in seeing an error about the top-level awaits in this library being incompatible with the built-target, or an error about await needing to only be in async functions or the top-level of a module (which it is!).

You may need to configure Vite to skip trying to optimize the walc.mjs file during bundling, something like:

export default defineConfig({

    // ..

    optimizeDeps: {
        exclude: [ "@lo-fi/webauthn-local-client" ]
    }

    // ..
});

SSR Breakage

An unfortunate gotcha of some tools that wrap Vite (e.g., Nuxt, etc) and do SSR (server-side rendering) is that they break a key assumption/behavior of this module's Vite plugin: the HTML injection of <script src="/walc-external-bundle.js"></script>.

As such, you'll likely need to manually add that <script> tag to your HTML pages/templates. The Vite plugin still copies that file into the public/ folder for you, so it should load once the tag is added to your HTML.

Webpack Plugin

If using Webpack 5+, make sure you're already using the HTML Webpack Plugin to manage building your index.html (and/or other HTML pages).

Then import this library's Webpack-plugin to manage the loading of its non-ESM dependencies. Add something like the following to your webpack.config.js:

// 'HtmlWebpackPlugin' is a required dependency of the
// local-vault Webpack plugin
import HtmlWebpackPlugin from "html-webpack-plugin";
import LV from "@lo-fi/local-vault/bundlers/webpack";

export default {
    // ..

    plugins: [
        // required LV dependency
        new HtmlWebpackPlugin({
            // ..
        }),

        LV()
    ],

    // ..
};

This plugin copies the @lo-fi/webauthn-local-client/dist/bundlers/walc-external-bundle.js file into the build root (default dist/), along with the other bundled files. It also injects a <script src="walc-external-bundle.js"></script> tag into the markup of the index.html file (and any other HTML files) that Webpack produces for your app.

Note: At present, this plugin is not configurable in any way (i.e., calling LV() above with no arguments). If something about its behavior is not compatible with your Webpack project setup -- which can vary widely and be quite complex to predict or support by a basic plugin -- it's recommended you simply copy over the @lo-fi/local-vault/bundler-plugins/webpack.mjs plugin and make necessary changes.

Import/Usage

To import and use local-vault in a bundled browser app:

// {WHICHEVER}: "idb", "local-storage", etc
import from "@lo-fi/local-vault/adapter/{WHICHEVER}";

import { connect } from "@lo-fi/local-vault";

When imported like this, Astro, Vite, Webpack should (via these plugins) properly find and bundle the @lo-fi/local-vault/dist/bundlers/lv.mjs ESM library module with the rest of your app code, hopefully without any further steps necessary.