Files
turso/bindings/javascript/src/lib.rs

769 lines
24 KiB
Rust

//! JavaScript bindings for the Turso library.
//!
//! These bindings provide a thin layer that exposes Turso's Rust API to JavaScript,
//! maintaining close alignment with the underlying implementation while offering
//! the following core database operations:
//!
//! - Opening and closing database connections
//! - Preparing SQL statements
//! - Binding parameters to prepared statements
//! - Iterating through query results
//! - Managing the I/O event loop
#[cfg(feature = "browser")]
pub mod browser;
use napi::bindgen_prelude::*;
use napi::{Env, Task};
use napi_derive::napi;
use std::sync::{Mutex, OnceLock};
use std::{
cell::{Cell, RefCell},
num::NonZeroUsize,
sync::Arc,
};
use tracing_subscriber::filter::LevelFilter;
use tracing_subscriber::fmt::format::FmtSpan;
use turso_core::storage::database::DatabaseFile;
/// Step result constants
const STEP_ROW: u32 = 1;
const STEP_DONE: u32 = 2;
const STEP_IO: u32 = 3;
/// The presentation mode for rows.
#[derive(Debug, Clone)]
enum PresentationMode {
Expanded,
Raw,
Pluck,
}
/// A database connection.
#[napi]
#[derive(Clone)]
pub struct Database {
inner: Option<Arc<DatabaseInner>>,
}
/// database inner is Send to the worker for initial connection
/// that's why we use OnceLock here - in order to make DatabaseInner Send and Sync
pub struct DatabaseInner {
path: String,
opts: Option<DatabaseOpts>,
io: Arc<dyn turso_core::IO>,
connect: OnceLock<DatabaseConnect>,
default_safe_integers: Mutex<bool>,
}
pub struct DatabaseConnect {
// hold db reference in order to keep it alive
// _db can be None if DB is controlled externally (for example, by sync-engine)
_db: Option<Arc<turso_core::Database>>,
conn: Arc<turso_core::Connection>,
}
pub(crate) fn is_memory(path: &str) -> bool {
path == ":memory:"
}
static TRACING_INIT: OnceLock<()> = OnceLock::new();
pub(crate) fn init_tracing(level_filter: &Option<String>) {
let Some(level_filter) = level_filter else {
return;
};
let level_filter = match level_filter.as_ref() {
"error" => LevelFilter::ERROR,
"warn" => LevelFilter::WARN,
"info" => LevelFilter::INFO,
"debug" => LevelFilter::DEBUG,
"trace" => LevelFilter::TRACE,
_ => return,
};
TRACING_INIT.get_or_init(|| {
tracing_subscriber::fmt()
.with_ansi(false)
.with_thread_ids(true)
.with_span_events(FmtSpan::ACTIVE)
.with_max_level(level_filter)
.init();
});
}
// for now we make DbTask unsound as turso_core::Database and turso_core::Connection are not fully thread-safe
unsafe impl Send for DbTask {}
pub enum DbTask {
Connect { db: Arc<DatabaseInner> },
}
impl Task for DbTask {
type Output = u32;
type JsValue = u32;
fn compute(&mut self) -> Result<Self::Output> {
match self {
DbTask::Connect { db } => {
connect_sync(db)?;
Ok(0)
}
}
}
fn resolve(&mut self, _: Env, output: Self::Output) -> Result<Self::JsValue> {
Ok(output)
}
}
/// Most of the options are aligned with better-sqlite API
/// (see https://github.com/WiseLibs/better-sqlite3/blob/master/docs/api.md#new-databasepath-options)
#[napi(object)]
#[derive(Clone)]
pub struct DatabaseOpts {
pub readonly: Option<bool>,
pub timeout: Option<u32>,
pub file_must_exist: Option<bool>,
pub tracing: Option<String>,
}
fn step_sync(stmt: &Arc<RefCell<turso_core::Statement>>) -> napi::Result<u32> {
let mut stmt = stmt.borrow_mut();
match stmt.step() {
Ok(turso_core::StepResult::Row) => Ok(STEP_ROW),
Ok(turso_core::StepResult::IO) => Ok(STEP_IO),
Ok(turso_core::StepResult::Done) => Ok(STEP_DONE),
Ok(turso_core::StepResult::Interrupt) => {
Err(create_generic_error("statement was interrupted"))
}
Ok(turso_core::StepResult::Busy) => Err(create_generic_error("database is busy")),
Err(e) => Err(to_generic_error("step failed", e)),
}
}
fn to_generic_error<E: std::error::Error>(message: &str, e: E) -> napi::Error {
Error::new(Status::GenericFailure, format!("{message}: {e}"))
}
fn to_error<E: std::error::Error>(status: napi::Status, message: &str, e: E) -> napi::Error {
Error::new(status, format!("{message}: {e}"))
}
fn create_generic_error(message: &str) -> napi::Error {
Error::new(Status::GenericFailure, message)
}
fn create_error(status: napi::Status, message: &str) -> napi::Error {
Error::new(status, message)
}
fn connect_sync(db: &DatabaseInner) -> napi::Result<()> {
if db.connect.get().is_some() {
return Ok(());
}
let mut flags = turso_core::OpenFlags::Create;
let mut busy_timeout = None;
if let Some(opts) = &db.opts {
if opts.readonly == Some(true) {
flags.set(turso_core::OpenFlags::ReadOnly, true);
flags.set(turso_core::OpenFlags::Create, false);
}
if opts.file_must_exist == Some(true) {
flags.set(turso_core::OpenFlags::Create, false);
}
if let Some(timeout) = opts.timeout {
busy_timeout = Some(std::time::Duration::from_millis(timeout as u64));
}
}
let io = &db.io;
let file = io
.open_file(&db.path, flags, false)
.map_err(|e| to_generic_error("failed to open file", e))?;
let db_file = DatabaseFile::new(file);
let db_core = turso_core::Database::open_with_flags(
io.clone(),
&db.path,
db_file,
flags,
turso_core::DatabaseOpts::new()
.with_mvcc(false)
.with_indexes(true),
None,
)
.map_err(|e| to_generic_error("failed to open database", e))?;
let conn = db_core
.connect()
.map_err(|e| to_generic_error("failed to connect", e))?;
if let Some(busy_timeout) = busy_timeout {
conn.set_busy_timeout(busy_timeout);
}
let connect = DatabaseConnect {
_db: Some(db_core),
conn,
};
// there can be races between concurrent connect - so let's ignore error in case of
let _ = db.connect.set(connect);
Ok(())
}
#[napi]
impl Database {
/// Creates a new database instance.
///
/// # Arguments
/// * `path` - The path to the database file.
#[napi(constructor)]
pub fn new(path: String, opts: Option<DatabaseOpts>) -> napi::Result<Self> {
let io: Arc<dyn turso_core::IO> = if is_memory(&path) {
Arc::new(turso_core::MemoryIO::new())
} else {
#[cfg(not(feature = "browser"))]
{
Arc::new(
turso_core::PlatformIO::new()
.map_err(|e| to_generic_error("failed to create IO", e))?,
)
}
#[cfg(feature = "browser")]
{
browser::opfs()
}
};
Self::new_with_io(path, io, opts)
}
pub fn new_with_io(
path: String,
io: Arc<dyn turso_core::IO>,
opts: Option<DatabaseOpts>,
) -> napi::Result<Self> {
if let Some(opts) = &opts {
init_tracing(&opts.tracing);
}
Ok(Self {
#[allow(clippy::arc_with_non_send_sync)]
inner: Some(Arc::new(DatabaseInner {
path,
opts,
io,
connect: OnceLock::new(),
default_safe_integers: Mutex::new(false),
})),
})
}
pub fn set_connected(&self, conn: Arc<turso_core::Connection>) -> napi::Result<()> {
let inner = self.inner()?;
inner
.connect
.set(DatabaseConnect { _db: None, conn })
.map_err(|_| create_generic_error("database was already connected"))?;
Ok(())
}
fn inner(&self) -> napi::Result<&Arc<DatabaseInner>> {
let Some(inner) = &self.inner else {
return Err(create_generic_error("database must be connected"));
};
Ok(inner)
}
/// Connect the database synchronously
/// This method is idempotent and can be called multiple times safely until the database will be closed
#[napi]
pub fn connect_sync(&self) -> napi::Result<()> {
connect_sync(self.inner()?)
}
/// Connect the database asynchronously
/// This method is idempotent and can be called multiple times safely until the database will be closed
#[napi(ts_return_type = "Promise<void>")]
pub fn connect_async(&self) -> napi::Result<AsyncTask<DbTask>> {
Ok(AsyncTask::new(DbTask::Connect {
db: self.inner()?.clone(),
}))
}
fn conn(&self) -> Result<Arc<turso_core::Connection>> {
let Some(DatabaseConnect { conn, .. }) = self.inner()?.connect.get() else {
return Err(create_generic_error("database must be connected"));
};
Ok(conn.clone())
}
/// Returns whether the database is in readonly-only mode.
#[napi(getter)]
pub fn readonly(&self) -> napi::Result<bool> {
Ok(self.conn()?.is_readonly(0))
}
/// Returns whether the database is in memory-only mode.
#[napi(getter)]
pub fn memory(&self) -> napi::Result<bool> {
Ok(is_memory(&self.inner()?.path))
}
/// Returns whether the database is in memory-only mode.
#[napi(getter)]
pub fn path(&self) -> napi::Result<String> {
Ok(self.inner()?.path.clone())
}
/// Returns whether the database connection is open.
#[napi(getter)]
pub fn open(&self) -> napi::Result<bool> {
if self.inner.is_none() {
return Ok(false);
}
Ok(self.inner()?.connect.get().is_some())
}
/// Prepares a statement for execution.
///
/// # Arguments
///
/// * `sql` - The SQL statement to prepare.
///
/// # Returns
///
/// A `Statement` instance.
#[napi]
pub fn prepare(&self, sql: String) -> napi::Result<Statement> {
let stmt = self
.conn()?
.prepare(&sql)
.map_err(|e| to_generic_error("prepare failed", e))?;
let column_names: Vec<std::ffi::CString> = (0..stmt.num_columns())
.map(|i| std::ffi::CString::new(stmt.get_column_name(i).to_string()).unwrap())
.collect();
Ok(Statement {
#[allow(clippy::arc_with_non_send_sync)]
stmt: Some(Arc::new(RefCell::new(stmt))),
column_names,
mode: RefCell::new(PresentationMode::Expanded),
safe_integers: Cell::new(*self.inner()?.default_safe_integers.lock().unwrap()),
})
}
#[napi]
pub fn executor(&self, sql: String) -> napi::Result<BatchExecutor> {
Ok(BatchExecutor {
conn: Some(self.conn()?.clone()),
sql,
position: 0,
stmt: None,
})
}
/// Returns the rowid of the last row inserted.
///
/// # Returns
///
/// The rowid of the last row inserted.
#[napi]
pub fn last_insert_rowid(&self) -> napi::Result<i64> {
Ok(self.conn()?.last_insert_rowid())
}
/// Returns the number of changes made by the last statement.
///
/// # Returns
///
/// The number of changes made by the last statement.
#[napi]
pub fn changes(&self) -> napi::Result<i64> {
Ok(self.conn()?.changes())
}
/// Returns the total number of changes made by all statements.
///
/// # Returns
///
/// The total number of changes made by all statements.
#[napi]
pub fn total_changes(&self) -> napi::Result<i64> {
Ok(self.conn()?.total_changes())
}
/// Closes the database connection.
///
/// # Returns
///
/// `Ok(())` if the database is closed successfully.
#[napi]
pub fn close(&mut self) -> napi::Result<()> {
let _ = self.inner.take();
Ok(())
}
/// Sets the default safe integers mode for all statements from this database.
///
/// # Arguments
///
/// * `toggle` - Whether to use safe integers by default.
#[napi(js_name = "defaultSafeIntegers")]
pub fn default_safe_integers(&self, toggle: Option<bool>) -> napi::Result<()> {
*self.inner()?.default_safe_integers.lock().unwrap() = toggle.unwrap_or(true);
Ok(())
}
/// Runs the I/O loop synchronously.
#[napi]
pub fn io_loop_sync(&self) -> napi::Result<()> {
let io = &self.inner()?.io;
io.step().map_err(|e| to_generic_error("IO error", e))?;
Ok(())
}
/// Runs the I/O loop asynchronously, returning a Promise.
#[napi(ts_return_type = "Promise<void>")]
pub fn io_loop_async(&self) -> napi::Result<AsyncTask<IoLoopTask>> {
let io = self.inner()?.io.clone();
Ok(AsyncTask::new(IoLoopTask { io }))
}
}
#[napi]
pub struct BatchExecutor {
conn: Option<Arc<turso_core::Connection>>,
sql: String,
position: usize,
stmt: Option<Arc<RefCell<turso_core::Statement>>>,
}
#[napi]
impl BatchExecutor {
#[napi]
pub fn step_sync(&mut self) -> Result<u32> {
loop {
if self.stmt.is_none() && self.position >= self.sql.len() {
return Ok(STEP_DONE);
}
if self.stmt.is_none() {
let conn = self.conn.as_ref().unwrap();
match conn.consume_stmt(&self.sql[self.position..]) {
#[allow(clippy::arc_with_non_send_sync)]
Ok(Some((stmt, offset))) => {
self.position += offset;
self.stmt = Some(Arc::new(RefCell::new(stmt)));
}
Ok(None) => return Ok(STEP_DONE),
Err(err) => return Err(to_generic_error("failed to consume stmt", err)),
}
}
let stmt = self.stmt.as_ref().unwrap();
match step_sync(stmt) {
Ok(STEP_DONE) => {
let _ = self.stmt.take();
continue;
}
result => return result,
}
}
}
#[napi]
pub fn reset(&mut self) {
let _ = self.conn.take();
let _ = self.stmt.take();
}
}
/// A prepared statement.
#[napi]
pub struct Statement {
stmt: Option<Arc<RefCell<turso_core::Statement>>>,
column_names: Vec<std::ffi::CString>,
mode: RefCell<PresentationMode>,
safe_integers: Cell<bool>,
}
#[napi]
impl Statement {
pub fn stmt(&self) -> napi::Result<&Arc<RefCell<turso_core::Statement>>> {
self.stmt
.as_ref()
.ok_or_else(|| create_generic_error("statement has been finalized"))
}
#[napi]
pub fn reset(&self) -> Result<()> {
self.stmt()?.borrow_mut().reset();
Ok(())
}
/// Returns the number of parameters in the statement.
#[napi]
pub fn parameter_count(&self) -> Result<u32> {
Ok(self.stmt()?.borrow().parameters_count() as u32)
}
/// Returns the name of a parameter at a specific 1-based index.
///
/// # Arguments
///
/// * `index` - The 1-based parameter index.
#[napi]
pub fn parameter_name(&self, index: u32) -> Result<Option<String>> {
let non_zero_idx = NonZeroUsize::new(index as usize).ok_or_else(|| {
create_error(Status::InvalidArg, "parameter index must be greater than 0")
})?;
let stmt = self.stmt()?.borrow();
Ok(stmt.parameters().name(non_zero_idx).map(|s| s.to_string()))
}
/// Binds a parameter at a specific 1-based index with explicit type.
///
/// # Arguments
///
/// * `index` - The 1-based parameter index.
/// * `value_type` - The type constant (0=null, 1=int, 2=float, 3=text, 4=blob).
/// * `value` - The value to bind.
#[napi]
pub fn bind_at(&self, index: u32, value: Unknown) -> Result<()> {
let non_zero_idx = NonZeroUsize::new(index as usize).ok_or_else(|| {
create_error(Status::InvalidArg, "parameter index must be greater than 0")
})?;
let value_type = value.get_type()?;
let turso_value = match value_type {
ValueType::Null => turso_core::Value::Null,
ValueType::Number => {
let n: f64 = unsafe { value.cast()? };
if n.fract() == 0.0 && n >= i64::MIN as f64 && n <= i64::MAX as f64 {
turso_core::Value::Integer(n as i64)
} else {
turso_core::Value::Float(n)
}
}
ValueType::BigInt => {
let bigint_str = value.coerce_to_string()?.into_utf8()?.as_str()?.to_owned();
let bigint_value = bigint_str
.parse::<i64>()
.map_err(|e| to_error(Status::NumberExpected, "failed to parse BigInt", e))?;
turso_core::Value::Integer(bigint_value)
}
ValueType::String => {
let s = value.coerce_to_string()?.into_utf8()?;
turso_core::Value::Text(s.as_str()?.to_owned().into())
}
ValueType::Boolean => {
let b: bool = unsafe { value.cast()? };
turso_core::Value::Integer(if b { 1 } else { 0 })
}
ValueType::Object => {
let obj = value.coerce_to_object()?;
if obj.is_buffer()? || obj.is_typedarray()? {
let length = obj.get_named_property::<u32>("length")?;
let mut bytes = Vec::with_capacity(length as usize);
for i in 0..length {
let byte = obj.get_element::<u32>(i)?;
bytes.push(byte as u8);
}
turso_core::Value::Blob(bytes)
} else {
let s = value.coerce_to_string()?.into_utf8()?;
turso_core::Value::Text(s.as_str()?.to_owned().into())
}
}
_ => {
let s = value.coerce_to_string()?.into_utf8()?;
turso_core::Value::Text(s.as_str()?.to_owned().into())
}
};
self.stmt()?.borrow_mut().bind_at(non_zero_idx, turso_value);
Ok(())
}
/// Step the statement and return result code (executed on the main thread):
/// 1 = Row available, 2 = Done, 3 = I/O needed
#[napi]
pub fn step_sync(&self) -> Result<u32> {
step_sync(self.stmt()?)
}
/// Get the current row data according to the presentation mode
#[napi]
pub fn row<'env>(&self, env: &'env Env) -> Result<Unknown<'env>> {
let stmt = self.stmt()?.borrow();
let row_data = stmt
.row()
.ok_or_else(|| create_generic_error("no row data available"))?;
let mode = self.mode.borrow();
let safe_integers = self.safe_integers.get();
let row_value = match *mode {
PresentationMode::Raw => {
let mut raw_array = env.create_array(row_data.len() as u32)?;
for (idx, value) in row_data.get_values().enumerate() {
let js_value = to_js_value(env, value, safe_integers)?;
raw_array.set(idx as u32, js_value)?;
}
raw_array.coerce_to_object()?.to_unknown()
}
PresentationMode::Pluck => {
let (_, value) =
row_data
.get_values()
.enumerate()
.next()
.ok_or(create_generic_error(
"pluck mode requires at least one column in the result",
))?;
to_js_value(env, value, safe_integers)?
}
PresentationMode::Expanded => {
let row = Object::new(env)?;
let raw_row = row.raw();
let raw_env = env.raw();
for idx in 0..row_data.len() {
let value = row_data.get_value(idx);
let column_name = &self.column_names[idx];
let js_value = to_js_value(env, value, safe_integers)?;
unsafe {
napi::sys::napi_set_named_property(
raw_env,
raw_row,
column_name.as_ptr(),
js_value.raw(),
);
}
}
row.to_unknown()
}
};
Ok(row_value)
}
/// Sets the presentation mode to raw.
#[napi]
pub fn raw(&mut self, raw: Option<bool>) {
self.mode = RefCell::new(match raw {
Some(false) => PresentationMode::Expanded,
_ => PresentationMode::Raw,
});
}
/// Sets the presentation mode to pluck.
#[napi]
pub fn pluck(&mut self, pluck: Option<bool>) {
self.mode = RefCell::new(match pluck {
Some(false) => PresentationMode::Expanded,
_ => PresentationMode::Pluck,
});
}
/// Sets safe integers mode for this statement.
///
/// # Arguments
///
/// * `toggle` - Whether to use safe integers.
#[napi(js_name = "safeIntegers")]
pub fn safe_integers(&self, toggle: Option<bool>) {
self.safe_integers.set(toggle.unwrap_or(true));
}
/// Get column information for the statement
#[napi(ts_return_type = "Promise<any>")]
pub fn columns<'env>(&self, env: &'env Env) -> Result<Array<'env>> {
let stmt = self.stmt()?.borrow();
let column_count = stmt.num_columns();
let mut js_array = env.create_array(column_count as u32)?;
for i in 0..column_count {
let mut js_obj = Object::new(env)?;
let column_name = stmt.get_column_name(i);
let column_type = stmt.get_column_type(i);
// Set the name property
js_obj.set("name", column_name.as_ref())?;
// Set type property if available
match column_type {
Some(type_str) => js_obj.set("type", type_str.as_str())?,
None => js_obj.set("type", ToNapiValue::into_unknown(Null, env)?)?,
}
// For now, set other properties to null since turso_core doesn't provide this metadata
js_obj.set("column", ToNapiValue::into_unknown(Null, env)?)?;
js_obj.set("table", ToNapiValue::into_unknown(Null, env)?)?;
js_obj.set("database", ToNapiValue::into_unknown(Null, env)?)?;
js_array.set(i as u32, js_obj)?;
}
Ok(js_array)
}
/// Finalizes the statement.
#[napi]
pub fn finalize(&mut self) -> Result<()> {
let _ = self.stmt.take();
Ok(())
}
}
/// Async task for running the I/O loop.
pub struct IoLoopTask {
// this field is public because it is also set in the sync package
pub io: Arc<dyn turso_core::IO>,
}
impl Task for IoLoopTask {
type Output = ();
type JsValue = ();
fn compute(&mut self) -> napi::Result<Self::Output> {
self.io
.step()
.map_err(|e| to_generic_error("IO error", e))?;
Ok(())
}
fn resolve(&mut self, _env: Env, _output: Self::Output) -> napi::Result<Self::JsValue> {
Ok(())
}
}
/// Convert a Turso value to a JavaScript value.
fn to_js_value<'a>(
env: &'a napi::Env,
value: &turso_core::Value,
safe_integers: bool,
) -> napi::Result<Unknown<'a>> {
match value {
turso_core::Value::Null => ToNapiValue::into_unknown(Null, env),
turso_core::Value::Integer(i) => {
if safe_integers {
let bigint = BigInt::from(*i);
ToNapiValue::into_unknown(bigint, env)
} else {
ToNapiValue::into_unknown(*i as f64, env)
}
}
turso_core::Value::Float(f) => ToNapiValue::into_unknown(*f, env),
turso_core::Value::Text(s) => ToNapiValue::into_unknown(s.as_str(), env),
turso_core::Value::Blob(b) => {
#[cfg(not(feature = "browser"))]
{
let buffer = Buffer::from(b.as_slice());
ToNapiValue::into_unknown(buffer, env)
}
// emnapi do not support Buffer
#[cfg(feature = "browser")]
{
let buffer = Uint8Array::from(b.as_slice());
ToNapiValue::into_unknown(buffer, env)
}
}
}
}