2 unstable releases

0.2.0 Feb 14, 2026
0.1.0 Feb 11, 2026

#276 in Procedural macros

MIT license

41KB
916 lines

oxichrome

License: MIT Rust Manifest V3 Website

oxichrome

Write browser extensions entirely in Rust, compiled to WebAssembly. Chrome and Firefox.


use leptos::prelude::*;
use oxichrome::prelude::*;

#[oxichrome::extension(
    name = "My Extension",
    version = "1.0.0",
    permissions = ["storage"]
)]
struct Extension;

#[oxichrome::background]
async fn start() {
    oxichrome::log!("Running!");
}

#[oxichrome::popup]
fn Popup() -> impl IntoView {
    view! { <p>"Hello from Rust."</p> }
}
cargo oxichrome build                  # Chrome/Edge/Brave
cargo oxichrome build --target firefox # Firefox

What it does

  • Five proc macros (#[extension], #[background], #[on], #[popup], #[options_page]) transform your Rust into wasm-bindgen exports
  • Typed bindings to chrome.storage, chrome.tabs, chrome.runtime
  • Leptos for reactive popup and options page UI
  • cargo oxichrome build handles everything: compilation, wasm-bindgen, manifest generation, JS/HTML shim generation, static asset copying, optional wasm-opt
  • --target firefox generates a Firefox-compatible Manifest V3 extension
  • Separate dist/chromium/ and dist/firefox/ output directories

Zero JavaScript written by hand.

Install

cargo install cargo-oxichrome

Requires:

  • Rust (stable)
  • wasm32-unknown-unknown target (auto-installed on first build)
  • wasm-bindgen-cli (auto-installed with version matching on first build)

Quick start

cargo oxichrome new my-extension
cd my-extension
cargo oxichrome build

cargo oxichrome new scaffolds a project with the oxichrome crate already in [dependencies]:

[dependencies]
oxichrome = { version = "0.2" }
wasm-bindgen = "0.2"
serde = { version = "1", features = ["derive"] }

Chrome/Edge: Load dist/chromium/ in chrome://extensions with "Load unpacked".

Firefox: Build with --target firefox, then load dist/firefox/manifest.json in about:debugging#/runtime/this-firefox.

CLI commands

Command Description
cargo oxichrome build Build for Chromium (default)
cargo oxichrome build --release Optimized release build
cargo oxichrome build --target firefox Build for Firefox
cargo oxichrome clean Remove the dist/ directory
cargo oxichrome new <name> Scaffold a new extension project

Project structure

oxichrome/
├── oxichrome/            re-export crate (what users depend on)
├── oxichrome-core/       runtime: Chrome API bindings, error types, logging
├── oxichrome-macros/     proc macros
├── oxichrome-build/      source parsing, manifest/shim generation
├── oxichrome-cli/        the cargo oxichrome command
└── examples/
    ├── counter-extension/
    └── color-picker/

Proc macros

#[oxichrome::extension(...)]

Defines extension metadata. Applied to a struct.

#[oxichrome::extension(
    name = "My Extension",
    version = "1.0.0",
    description = "Optional description",
    permissions = ["storage", "tabs"]
)]
struct MyExtension;

#[oxichrome::background]

Marks an async function as the background service worker entry point.

#[oxichrome::background]
async fn start() {
    oxichrome::log!("Started.");
}

#[oxichrome::on(namespace::event)]

Registers an async function as a Chrome event listener.

#[oxichrome::on(runtime::on_installed)]
async fn on_install(details: oxichrome::__private::wasm_bindgen::JsValue) {
    oxichrome::log!("Installed: {:?}", details);
}

#[oxichrome::popup]

Marks a Leptos component as the popup UI.

#[oxichrome::popup]
fn Popup() -> impl IntoView {
    view! { <p>"Hello."</p> }
}

#[oxichrome::options_page]

Marks a Leptos component as the options page UI.

#[oxichrome::options_page]
fn Options() -> impl IntoView {
    view! { <h1>"Settings"</h1> }
}

Chrome APIs

// Storage
let val: Option<i32> = oxichrome::storage::get("key").await?;
oxichrome::storage::set("key", &42).await?;
oxichrome::storage::remove("key").await?;

// Tabs
let tabs: Vec<Tab> = oxichrome::tabs::query(&query).await?;
let tab: Tab = oxichrome::tabs::create(&props).await?;
oxichrome::tabs::send_message(tab_id, &msg).await?;

// Runtime
let url = oxichrome::runtime::get_url("icon.png");
oxichrome::runtime::send_message(&msg).await?;

Build output

dist/
├── chromium/                  # default target
│   ├── manifest.json
│   ├── background.js
│   ├── popup.html / popup.js
│   ├── options.html / options.js
│   └── wasm/
│       ├── crate_name.js
│       └── crate_name_bg.wasm
└── firefox/                   # --target firefox
    ├── manifest.json          # background scripts, gecko ID
    └── ...                    # same files as chromium/

License

MIT

Dependencies

~1.3–2.5MB
~46K SLoC