No description
Find a file
2024-04-18 00:14:00 +08:00
.cargo fix(napi): compile error for wasm32-unknown-unknown target 2023-11-20 17:10:58 +08:00
.github ci: fix flaky tests (#2033) 2024-04-10 15:43:05 +08:00
.husky fix(napi-derive): JsArrayBuffer generated type 2024-01-26 14:32:24 +08:00
.yarn chore(deps): update yarn to v4.1.1 (#1989) 2024-03-05 11:31:06 +08:00
bench fix(napi): External should impl FromNapiRef rather than FromNapiValue (#2013) 2024-03-25 15:11:11 +08:00
cli chore(release): publish 2024-04-17 20:23:19 +08:00
crates fix(napi-derive): bail the unexpected factory directive (#2051) 2024-04-18 00:14:00 +08:00
examples fix(napi-derive): bail the unexpected factory directive (#2051) 2024-04-18 00:14:00 +08:00
images docs: add next.js into README 2021-07-30 13:14:54 +08:00
memory-testing feat(napi): support Return generic of ThreadsafeFunction (#1997) 2024-03-20 21:37:08 +08:00
triples chore(release): publish 2024-04-17 20:23:19 +08:00
wasm-runtime chore(release): publish 2024-03-21 15:08:50 +08:00
.cirrus.yml ci: fix arm64 job (#1998) 2024-03-10 20:41:55 +08:00
.dockerignore feat(napi): support musl linux 2020-06-11 16:20:37 +08:00
.editorconfig init 2018-04-28 16:26:38 +08:00
.gitattributes chore: upgrade to yarn3 2022-01-24 17:25:40 +08:00
.gitignore chore: add .idea folder to ignore (#1801) 2023-11-13 10:07:50 +08:00
.npmignore feat: support linux aarch64 2020-10-15 09:12:43 +08:00
.oxlintignore chore: use oxlint (#2032) 2024-04-10 14:26:34 +08:00
.prettierignore chore: upgrade emnapi dependencies (#1817) 2023-11-19 15:13:06 +08:00
.yarnrc.yml chore(deps): update yarn to v4.1.1 (#1989) 2024-03-05 11:31:06 +08:00
alpine-zig.Dockerfile build: upgrade toolchain in Docker (#1810) 2023-11-14 09:58:58 +08:00
alpine.Dockerfile build: update Node.js version in alpine.Dockerfile 2023-03-31 14:19:05 +08:00
Cargo.toml refactor(cli): --profile flag (#1604) 2023-05-27 12:00:43 +08:00
CODE_OF_CONDUCT.md doc: add email to CODE_OF_CONDUCT 2020-05-20 18:49:36 +08:00
debian-aarch64.Dockerfile ci: upgrade toolchain in Docker (#1956) 2024-02-18 13:42:10 +08:00
debian-zig.Dockerfile build: upgrade toolchain in Docker (#1810) 2023-11-14 09:58:58 +08:00
debian.Dockerfile ci: upgrade toolchain in Docker (#1956) 2024-02-18 13:42:10 +08:00
lerna.json ci: add release npm cli 2023-07-18 11:06:45 +08:00
LICENSE docs: update license 2021-01-25 14:12:44 +08:00
package.json chore(deps): update dependency electron to v30 (#2043) 2024-04-16 14:17:22 +08:00
README.md chore: drop node 17, add node 18 2022-04-27 13:17:39 +08:00
renovate.json chore: fix renovate path and dedupe electron versions (#1974) 2024-02-25 00:50:22 +08:00
rustfmt.toml feat(cli): refactor cli build 2021-11-19 18:22:40 +08:00
tsconfig.json fix: prevent crashing when napi_register_module_v1 is called twice (#1554) 2023-04-08 23:08:48 +08:00
tsconfig.root-lint.json feat: add wasm runtime package (#1904) 2024-01-16 23:28:40 +08:00
yarn.lock feat(cli): optimize wasm output binary (#2049) 2024-04-17 20:13:02 +08:00

napi-rs

This project was initialized from xray

A framework for building compiled Node.js add-ons in Rust via Node-API. Website: https://napi.rs

chat Stake to support us

Platform Support

Test & Release FreeBSD Address Sanitizer Memory Leak Detect

MSRV

Rust 1.65.0

node12 node14 node16 node18 node20
Windows x64
Windows x86
Windows arm64
macOS x64
macOS aarch64
Linux x64 gnu
Linux x64 musl
Linux aarch64 gnu
Linux aarch64 musl
Linux arm gnueabihf
Linux arm muslebihf
Linux powerpc64le gnu
Linux s390x gnu
Linux riscv64 gnu N/A N/A
Linux aarch64 android
Linux armv7 android
FreeBSD x64

This library depends on Node-API and requires Node@10.0.0 or later.

We already have some packages written by napi-rs: node-rs

One nice feature is that this crate allows you to build add-ons purely with the Rust/JavaScript toolchain and without involving node-gyp.

Taste

You can start from package-template to play with napi-rs

Define JavaScript functions

/// import the preludes
use napi::bindgen_prelude::*;
use napi_derive::napi;

/// module registration is done by the runtime, no need to explicitly do it now.
#[napi]
fn fibonacci(n: u32) -> u32 {
  match n {
    1 | 2 => 1,
    _ => fibonacci(n - 1) + fibonacci(n - 2),
  }
}

/// use `Fn`, `FnMut` or `FnOnce` traits to defined JavaScript callbacks
/// the return type of callbacks can only be `Result`.
#[napi]
fn get_cwd<T: Fn(String) -> Result<()>>(callback: T) {
  callback(env::current_dir().unwrap().to_string_lossy().to_string()).unwrap();
}

/// or, define the callback signature in where clause
#[napi]
fn test_callback<T>(callback: T)
where T: Fn(String) -> Result<()>
{}

/// async fn, require `async` feature enabled.
/// [dependencies]
/// napi = {version="2", features=["async"]}
#[napi]
async fn read_file_async(path: String) -> Result<Buffer> {
  tokio::fs::read(path)
    .map(|r| match r {
      Ok(content) => Ok(content.into()),
      Err(e) => Err(Error::new(
        Status::GenericFailure,
        format!("failed to read file, {}", e),
      )),
    })
    .await
}

more examples at examples

Building

This repository is a Cargo crate. Any napi-based add-on should contain Cargo.toml to make it a Cargo crate.

In your Cargo.toml you need to set the crate-type to "cdylib" so that cargo builds a C-style shared library that can be dynamically loaded by the Node executable. You'll also need to add this crate as a dependency.

[package]
name = "awesome"

[lib]
crate-type = ["cdylib"]

[dependencies]
napi = "3"
napi-derive = "3"

[build-dependencies]
napi-build = "1"

And create build.rs in your own project:

// build.rs
extern crate napi_build;

fn main() {
  napi_build::setup();
}

So far, the napi build script has only been tested on macOS Linux Windows x64 MSVC and FreeBSD.

Install the @napi-rs/cli to help you build your Rust codes and copy Dynamic lib file to .node file in case you can require it in your program.

{
  "package": "awesome-package",
  "devDependencies": {
    "@napi-rs/cli": "^1.0.0"
  },
  "napi": {
    "name": "jarvis" // <----------- Config the name of native addon, or the napi command will use the name of `Cargo.toml` for the binary file name.
  },
  "scripts": {
    "build": "napi build --release",
    "build:debug": "napi build"
  }
}

Then you can require your native binding:

require('./jarvis.node')

The module_name would be your package name in your Cargo.toml.

xxx => ./xxx.node

xxx-yyy => ./xxx_yyy.node

You can also copy Dynamic lib file to an appointed location:

napi build [--release] ./dll
napi build [--release] ./artifacts

There are documents which contains more details about the @napi-rs/cli usage.

Testing

Because libraries that depend on this crate must be loaded into a Node executable in order to resolve symbols, all tests are written in JavaScript in the test_module subdirectory.

To run tests:

yarn build:test
yarn test

Features table

Rust Type Node Type NAPI Version Minimal Node version Enable by napi feature
u32 Number 1 v8.0.0
i32/i64 Number 1 v8.0.0
f64 Number 1 v8.0.0
bool Boolean 1 v8.0.0
String/&'a str String 1 v8.0.0
Latin1String String 1 v8.0.0 latin1
UTF16String String 1 v8.0.0
Object Object 1 v8.0.0
serde_json::Map Object 1 v8.0.0 serde-json
serde_json::Value any 1 v8.0.0 serde-json
Array Array 1 v8.0.0
Vec Array 1 v8.0.0
Buffer Buffer 1 v8.0.0
External External 1 v8.0.0
Null null 1 v8.0.0
Undefined/() undefined 1 v8.0.0
Result<()> Error 1 v8.0.0
T: Fn(...) -> Result Function 1 v8.0.0
Async/Future Promise 4 v10.6.0 async
AsyncTask Promise 1 v8.5.0
JsGlobal global 1 v8.0.0
JsSymbol Symbol 1 v8.0.0
Int8Array/Uint8Array ... TypedArray 1 v8.0.0
JsFunction threadsafe function 4 v10.6.0 napi4
BigInt BigInt 6 v10.7.0 napi6