mirror of
https://github.com/lancedb/lancedb.git
synced 2026-01-10 22:02:58 +00:00
This exposes the `LANCEDB_LOG` environment variable in node, so that users can now turn on logging. In addition, fixes a bug where only the top-level error from Rust was being shown. This PR makes sure the full error chain is included in the error message. In the future, will improve this so the error chain is set on the [cause](https://nodejs.org/api/errors.html#errorcause) property of JS errors https://github.com/lancedb/lancedb/issues/1779 Fixes #1774
224 lines
7.5 KiB
Rust
224 lines
7.5 KiB
Rust
// Copyright 2024 Lance Developers.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
use std::collections::HashMap;
|
|
use std::str::FromStr;
|
|
|
|
use napi::bindgen_prelude::*;
|
|
use napi_derive::*;
|
|
|
|
use crate::error::{convert_error, NapiErrorExt};
|
|
use crate::table::Table;
|
|
use crate::ConnectionOptions;
|
|
use lancedb::connection::{
|
|
ConnectBuilder, Connection as LanceDBConnection, CreateTableMode, LanceFileVersion,
|
|
};
|
|
use lancedb::ipc::{ipc_file_to_batches, ipc_file_to_schema};
|
|
|
|
#[napi]
|
|
pub struct Connection {
|
|
inner: Option<LanceDBConnection>,
|
|
}
|
|
|
|
impl Connection {
|
|
pub(crate) fn inner_new(inner: LanceDBConnection) -> Self {
|
|
Self { inner: Some(inner) }
|
|
}
|
|
|
|
fn get_inner(&self) -> napi::Result<&LanceDBConnection> {
|
|
self.inner
|
|
.as_ref()
|
|
.ok_or_else(|| napi::Error::from_reason("Connection is closed"))
|
|
}
|
|
}
|
|
|
|
impl Connection {
|
|
fn parse_create_mode_str(mode: &str) -> napi::Result<CreateTableMode> {
|
|
match mode {
|
|
"create" => Ok(CreateTableMode::Create),
|
|
"overwrite" => Ok(CreateTableMode::Overwrite),
|
|
"exist_ok" => Ok(CreateTableMode::exist_ok(|builder| builder)),
|
|
_ => Err(napi::Error::from_reason(format!("Invalid mode {}", mode))),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[napi]
|
|
impl Connection {
|
|
/// Create a new Connection instance from the given URI.
|
|
#[napi(factory)]
|
|
pub async fn new(uri: String, options: ConnectionOptions) -> napi::Result<Self> {
|
|
let mut builder = ConnectBuilder::new(&uri);
|
|
if let Some(interval) = options.read_consistency_interval {
|
|
builder =
|
|
builder.read_consistency_interval(std::time::Duration::from_secs_f64(interval));
|
|
}
|
|
if let Some(storage_options) = options.storage_options {
|
|
for (key, value) in storage_options {
|
|
builder = builder.storage_option(key, value);
|
|
}
|
|
}
|
|
|
|
let client_config = options.client_config.unwrap_or_default();
|
|
builder = builder.client_config(client_config.into());
|
|
|
|
if let Some(api_key) = options.api_key {
|
|
builder = builder.api_key(&api_key);
|
|
}
|
|
|
|
if let Some(region) = options.region {
|
|
builder = builder.region(®ion);
|
|
} else {
|
|
builder = builder.region("us-east-1");
|
|
}
|
|
|
|
if let Some(host_override) = options.host_override {
|
|
builder = builder.host_override(&host_override);
|
|
}
|
|
|
|
Ok(Self::inner_new(builder.execute().await.default_error()?))
|
|
}
|
|
|
|
#[napi]
|
|
pub fn display(&self) -> napi::Result<String> {
|
|
Ok(self.get_inner()?.to_string())
|
|
}
|
|
|
|
#[napi]
|
|
pub fn is_open(&self) -> bool {
|
|
self.inner.is_some()
|
|
}
|
|
|
|
#[napi]
|
|
pub fn close(&mut self) {
|
|
self.inner.take();
|
|
}
|
|
|
|
/// List all tables in the dataset.
|
|
#[napi(catch_unwind)]
|
|
pub async fn table_names(
|
|
&self,
|
|
start_after: Option<String>,
|
|
limit: Option<u32>,
|
|
) -> napi::Result<Vec<String>> {
|
|
let mut op = self.get_inner()?.table_names();
|
|
if let Some(start_after) = start_after {
|
|
op = op.start_after(start_after);
|
|
}
|
|
if let Some(limit) = limit {
|
|
op = op.limit(limit);
|
|
}
|
|
op.execute().await.default_error()
|
|
}
|
|
|
|
/// Create table from a Apache Arrow IPC (file) buffer.
|
|
///
|
|
/// Parameters:
|
|
/// - name: The name of the table.
|
|
/// - buf: The buffer containing the IPC file.
|
|
///
|
|
#[napi(catch_unwind)]
|
|
pub async fn create_table(
|
|
&self,
|
|
name: String,
|
|
buf: Buffer,
|
|
mode: String,
|
|
storage_options: Option<HashMap<String, String>>,
|
|
data_storage_options: Option<String>,
|
|
enable_v2_manifest_paths: Option<bool>,
|
|
) -> napi::Result<Table> {
|
|
let batches = ipc_file_to_batches(buf.to_vec())
|
|
.map_err(|e| napi::Error::from_reason(format!("Failed to read IPC file: {}", e)))?;
|
|
let mode = Self::parse_create_mode_str(&mode)?;
|
|
let mut builder = self.get_inner()?.create_table(&name, batches).mode(mode);
|
|
|
|
if let Some(storage_options) = storage_options {
|
|
for (key, value) in storage_options {
|
|
builder = builder.storage_option(key, value);
|
|
}
|
|
}
|
|
if let Some(data_storage_option) = data_storage_options.as_ref() {
|
|
builder = builder.data_storage_version(
|
|
LanceFileVersion::from_str(data_storage_option).map_err(|e| convert_error(&e))?,
|
|
);
|
|
}
|
|
if let Some(enable_v2_manifest_paths) = enable_v2_manifest_paths {
|
|
builder = builder.enable_v2_manifest_paths(enable_v2_manifest_paths);
|
|
}
|
|
let tbl = builder.execute().await.default_error()?;
|
|
Ok(Table::new(tbl))
|
|
}
|
|
|
|
#[napi(catch_unwind)]
|
|
pub async fn create_empty_table(
|
|
&self,
|
|
name: String,
|
|
schema_buf: Buffer,
|
|
mode: String,
|
|
storage_options: Option<HashMap<String, String>>,
|
|
data_storage_options: Option<String>,
|
|
enable_v2_manifest_paths: Option<bool>,
|
|
) -> napi::Result<Table> {
|
|
let schema = ipc_file_to_schema(schema_buf.to_vec()).map_err(|e| {
|
|
napi::Error::from_reason(format!("Failed to marshal schema from JS to Rust: {}", e))
|
|
})?;
|
|
let mode = Self::parse_create_mode_str(&mode)?;
|
|
let mut builder = self
|
|
.get_inner()?
|
|
.create_empty_table(&name, schema)
|
|
.mode(mode);
|
|
if let Some(storage_options) = storage_options {
|
|
for (key, value) in storage_options {
|
|
builder = builder.storage_option(key, value);
|
|
}
|
|
}
|
|
if let Some(data_storage_option) = data_storage_options.as_ref() {
|
|
builder = builder.data_storage_version(
|
|
LanceFileVersion::from_str(data_storage_option).map_err(|e| convert_error(&e))?,
|
|
);
|
|
}
|
|
if let Some(enable_v2_manifest_paths) = enable_v2_manifest_paths {
|
|
builder = builder.enable_v2_manifest_paths(enable_v2_manifest_paths);
|
|
}
|
|
let tbl = builder.execute().await.default_error()?;
|
|
Ok(Table::new(tbl))
|
|
}
|
|
|
|
#[napi(catch_unwind)]
|
|
pub async fn open_table(
|
|
&self,
|
|
name: String,
|
|
storage_options: Option<HashMap<String, String>>,
|
|
index_cache_size: Option<u32>,
|
|
) -> napi::Result<Table> {
|
|
let mut builder = self.get_inner()?.open_table(&name);
|
|
if let Some(storage_options) = storage_options {
|
|
for (key, value) in storage_options {
|
|
builder = builder.storage_option(key, value);
|
|
}
|
|
}
|
|
if let Some(index_cache_size) = index_cache_size {
|
|
builder = builder.index_cache_size(index_cache_size);
|
|
}
|
|
let tbl = builder.execute().await.default_error()?;
|
|
Ok(Table::new(tbl))
|
|
}
|
|
|
|
/// Drop table with the name. Or raise an error if the table does not exist.
|
|
#[napi(catch_unwind)]
|
|
pub async fn drop_table(&self, name: String) -> napi::Result<()> {
|
|
self.get_inner()?.drop_table(&name).await.default_error()
|
|
}
|
|
}
|