AgentSkillsCN

napi-rs-node-bindings

通过 napi-rs 为 Rust 库提供 Node.js 绑定。适用于开发 Node.js/JavaScript 与 Rust 代码的绑定项目,包括创建或修改 napi 宏与结构体、在 Rust 与 JavaScript 之间进行类型转换、跨 FFI 边界处理错误、运用 async/Promise 模式、构建原生模块、发布至 npm、测试绑定代码,或调试绑定相关问题。该方案专为 CQLite(Cassandra CQL 绑定)打造,并同步追踪功能一致性。

SKILL.md
--- frontmatter
name: napi-rs-node-bindings
description: "Node.js bindings for Rust libraries using napi-rs. Use when working on Node.js/JavaScript bindings to Rust code, including creating or modifying napi macros and structs, converting types between Rust and JavaScript, handling errors across the FFI boundary, async/Promise patterns, building native modules, publishing to npm, testing binding code, or debugging binding issues. Tuned for CQLite (Cassandra CQL bindings) with feature parity tracking."

napi-rs Node.js Bindings

Project Structure

code
my-rust-lib/
├── Cargo.toml          # napi dependency
├── package.json        # npm package config
├── src/
│   ├── lib.rs          # Core Rust library
│   └── node/           # Node binding module
│       ├── mod.rs      # Exports
│       ├── types.rs    # #[napi] structs
│       └── errors.rs   # Error conversions
├── index.js            # JS entry point (generated)
├── index.d.ts          # TypeScript definitions (generated)
└── __test__/
    └── index.spec.ts   # Tests

Core Workflow

1. Expose Rust Types

rust
use napi::bindgen_prelude::*;
use napi_derive::napi;

#[napi]
pub struct MyType {
    inner: RustType,  // Private Rust type
}

#[napi]
impl MyType {
    #[napi(constructor)]
    pub fn new(value: i64) -> Result<Self> {
        Ok(Self { inner: RustType::new(value)? })
    }
    
    #[napi]
    pub fn process(&self) -> Result<String> {
        self.inner.process().map_err(|e| e.into())
    }
}

2. Export Functions

rust
#[napi]
pub fn parse_cql(query: String) -> Result<Statement> {
    cqlite::parse(&query).map_err(|e| e.into())
}

// Async function returns Promise
#[napi]
pub async fn parse_file(path: String) -> Result<Vec<Statement>> {
    let content = tokio::fs::read_to_string(&path).await?;
    cqlite::parse_all(&content).map_err(|e| e.into())
}

3. Build & Test

bash
npm run build          # Build native module
npm run build:debug    # Debug build (faster)
npm test               # Run tests

Reference Guides

Load these as needed based on the task:

TaskReference
Type mapping between Rust ↔ JavaScripttype-conversions.md
Converting Rust errors to JS exceptionserror-handling.md
Async patterns and Promisesasync-patterns.md
Building and publishing to npmbuild-publish.md
Testing strategiestesting.md
Debugging common binding issuesdebugging.md
CQLite CQL feature parity checklistcqlite-parity.md

Quick Reference

Cargo.toml Setup

toml
[package]
name = "cqlite"
version = "0.1.0"
edition = "2021"

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

[dependencies]
napi = { version = "2", default-features = false, features = ["napi9", "async", "serde-json"] }
napi-derive = "2"

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

[profile.release]
lto = true

package.json Setup

json
{
  "name": "cqlite",
  "version": "0.1.0",
  "main": "index.js",
  "types": "index.d.ts",
  "napi": {
    "name": "cqlite",
    "triples": {
      "defaults": true,
      "additional": ["aarch64-apple-darwin", "aarch64-linux-android"]
    }
  },
  "scripts": {
    "build": "napi build --platform --release",
    "build:debug": "napi build --platform",
    "prepublishOnly": "napi prepublish -t npm",
    "test": "ava"
  },
  "devDependencies": {
    "@napi-rs/cli": "^2.18.0",
    "ava": "^6.0.0"
  }
}

build.rs

rust
extern crate napi_build;

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

napi-rs Attribute Quick Reference

AttributeUse
#[napi]Expose function/struct/impl to JS
#[napi(constructor)]Class constructor
#[napi(getter)]Property getter
#[napi(setter)]Property setter
#[napi(factory)]Static factory method
#[napi(js_name = "...")]Rename in JavaScript
#[napi(ts_args_type = "...")]Custom TypeScript arg types
#[napi(ts_return_type = "...")]Custom TypeScript return type
#[napi(object)]Plain JS object (no class)
#[napi(strict)]Strict type checking