mirror of
https://github.com/aljazceru/turso.git
synced 2026-01-03 00:14:21 +01:00
bind/js: Partially implements iterate() method
The API still is sync and isn't variadic
This commit is contained in:
@@ -16,4 +16,4 @@ napi = { version = "2.16.17", default-features = false, features = ["napi4"] }
|
||||
napi-derive = { version = "2.16.13", default-features = false }
|
||||
|
||||
[build-dependencies]
|
||||
napi-build = "2.0.1"
|
||||
napi-build = "2.2.0"
|
||||
|
||||
@@ -7,7 +7,6 @@ test("Open in-memory database", async (t) => {
|
||||
t.is(db.memory, true);
|
||||
});
|
||||
|
||||
|
||||
test("Statement.get() returns data", async (t) => {
|
||||
const [db] = await connect(":memory:");
|
||||
const stmt = db.prepare("SELECT 1");
|
||||
@@ -31,11 +30,26 @@ test("Statement.run() returns correct result object", async (t) => {
|
||||
t.deepEqual(rows, { changes: 1, lastInsertRowid: 1 });
|
||||
});
|
||||
|
||||
test("Statment.iterate() should correctly return an iterable object", async (t) => {
|
||||
const [db] = await connect(":memory:");
|
||||
db.prepare("CREATE TABLE users (name TEXT, age INTEGER)").run();
|
||||
db.prepare("INSERT INTO users (name, age) VALUES (?, ?)").run("Alice", 42);
|
||||
db.prepare("INSERT INTO users (name, age) VALUES (?, ?)").run("Bob", 24);
|
||||
let rows = db.prepare("SELECT * FROM users").iterate();
|
||||
for (const row of rows) {
|
||||
t.truthy(row.name);
|
||||
t.true(typeof row.age === "number");
|
||||
}
|
||||
});
|
||||
|
||||
test("Empty prepared statement should throw", async (t) => {
|
||||
const [db] = await connect(":memory:");
|
||||
t.throws(() => {
|
||||
db.prepare("");
|
||||
}, { instanceOf: Error });
|
||||
const [db] = await connect(":memory:");
|
||||
t.throws(
|
||||
() => {
|
||||
db.prepare("");
|
||||
},
|
||||
{ instanceOf: Error },
|
||||
);
|
||||
});
|
||||
|
||||
const connect = async (path) => {
|
||||
|
||||
@@ -27,10 +27,22 @@ test("Statement.get() returns null when no data", async (t) => {
|
||||
// it should return a result object, not a row object
|
||||
test("Statement.run() returns correct result object", async (t) => {
|
||||
const [db] = await connect(":memory:");
|
||||
db.prepare("CREATE TABLE users (name TEXT)").run();
|
||||
db.prepare("INSERT INTO users (name) VALUES (?)").run(["Alice"]);
|
||||
db.prepare("CREATE TABLE users (name TEXT, age INTEGER)").run();
|
||||
db.prepare("INSERT INTO users (name, age) VALUES (?, ?)").run(["Alice", 42]);
|
||||
let rows = db.prepare("SELECT * FROM users").all();
|
||||
t.deepEqual(rows, [{ name: "Alice" }]);
|
||||
t.deepEqual(rows, [{ name: "Alice", age: 42 }]);
|
||||
});
|
||||
|
||||
test("Statment.iterate() should correctly return an iterable object", async (t) => {
|
||||
const [db] = await connect(":memory:");
|
||||
db.prepare("CREATE TABLE users (name TEXT, age INTEGER)").run();
|
||||
db.prepare("INSERT INTO users (name, age) VALUES (?, ?)").run(["Alice", 42]);
|
||||
db.prepare("INSERT INTO users (name, age) VALUES (?, ?)").run(["Bob", 24]);
|
||||
let rows = db.prepare("SELECT * FROM users").iterate();
|
||||
for (const row of rows) {
|
||||
t.truthy(row.name);
|
||||
t.true(typeof row.age === "number");
|
||||
}
|
||||
});
|
||||
|
||||
test("Empty prepared statement should throw", async (t) => {
|
||||
|
||||
@@ -8,6 +8,8 @@ use std::sync::Arc;
|
||||
|
||||
use limbo_core::types::Text;
|
||||
use limbo_core::{maybe_init_database_file, LimboError};
|
||||
use napi::iterator::Generator;
|
||||
use napi::JsObject;
|
||||
use napi::{bindgen_prelude::ObjectFinalize, Env, JsUnknown};
|
||||
use napi_derive::napi;
|
||||
|
||||
@@ -136,20 +138,23 @@ impl Database {
|
||||
#[napi]
|
||||
pub struct Statement {
|
||||
// TODO: implement each property when core supports it
|
||||
// #[napi(writable = false)]
|
||||
// #[napi(able = false)]
|
||||
// pub reader: bool,
|
||||
// #[napi(writable = false)]
|
||||
// pub readonly: bool,
|
||||
// #[napi(writable = false)]
|
||||
// pub busy: bool,
|
||||
database: Database,
|
||||
inner: RefCell<limbo_core::Statement>,
|
||||
inner: Rc<RefCell<limbo_core::Statement>>,
|
||||
}
|
||||
|
||||
#[napi]
|
||||
impl Statement {
|
||||
pub fn new(inner: RefCell<limbo_core::Statement>, database: Database) -> Self {
|
||||
Self { inner, database }
|
||||
Self {
|
||||
inner: Rc::new(inner),
|
||||
database,
|
||||
}
|
||||
}
|
||||
|
||||
#[napi]
|
||||
@@ -194,8 +199,12 @@ impl Statement {
|
||||
}
|
||||
|
||||
#[napi]
|
||||
pub fn iterate() {
|
||||
todo!()
|
||||
pub fn iterate(&self, env: Env) -> IteratorStatement {
|
||||
IteratorStatement {
|
||||
stmt: Rc::clone(&self.inner),
|
||||
database: self.database.clone(),
|
||||
env,
|
||||
}
|
||||
}
|
||||
|
||||
#[napi]
|
||||
@@ -266,6 +275,46 @@ impl Statement {
|
||||
}
|
||||
}
|
||||
|
||||
#[napi(iterator)]
|
||||
pub struct IteratorStatement {
|
||||
stmt: Rc<RefCell<limbo_core::Statement>>,
|
||||
database: Database,
|
||||
env: Env,
|
||||
}
|
||||
|
||||
impl Generator for IteratorStatement {
|
||||
type Yield = JsObject;
|
||||
|
||||
type Next = ();
|
||||
|
||||
type Return = ();
|
||||
|
||||
fn next(&mut self, _: Option<Self::Next>) -> Option<Self::Yield> {
|
||||
let mut stmt = self.stmt.borrow_mut();
|
||||
|
||||
match stmt.step().ok()? {
|
||||
limbo_core::StepResult::Row => {
|
||||
let row = stmt.row().unwrap();
|
||||
let mut js_row = self.env.create_object().ok()?;
|
||||
|
||||
for (idx, value) in row.get_values().enumerate() {
|
||||
let key = stmt.get_column_name(idx);
|
||||
let js_value = to_js_value(&self.env, value);
|
||||
js_row.set_named_property(&key, js_value).ok()?;
|
||||
}
|
||||
|
||||
Some(js_row)
|
||||
}
|
||||
limbo_core::StepResult::Done => None,
|
||||
limbo_core::StepResult::IO => {
|
||||
self.database.io.run_once().ok()?;
|
||||
None // clearly it's incorrect it should return to user
|
||||
}
|
||||
limbo_core::StepResult::Interrupt | limbo_core::StepResult::Busy => None,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn to_js_value(env: &napi::Env, value: &limbo_core::Value) -> napi::Result<JsUnknown> {
|
||||
match value {
|
||||
limbo_core::Value::Null => Ok(env.get_null()?.into_unknown()),
|
||||
|
||||
Reference in New Issue
Block a user