mirror of
https://github.com/lancedb/lancedb.git
synced 2026-01-09 21:32:58 +00:00
- Rename safe_import -> attempt_import_or_raise (closes https://github.com/lancedb/lancedb/pull/923) - Update docs - Add Notebook example (@changhiskhan you can use it for the talk. Comes with "open in colab" button) - Latency benchmark & results comparison, sanity check on real-world data - Updates the default openai model to gpt-4
242 lines
6.2 KiB
Python
242 lines
6.2 KiB
Python
# Copyright 2023 LanceDB 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.
|
|
|
|
import importlib
|
|
import os
|
|
import pathlib
|
|
from datetime import date, datetime
|
|
from functools import singledispatch
|
|
from typing import Tuple, Union
|
|
from urllib.parse import urlparse
|
|
|
|
import numpy as np
|
|
import pyarrow as pa
|
|
import pyarrow.fs as pa_fs
|
|
|
|
|
|
def get_uri_scheme(uri: str) -> str:
|
|
"""
|
|
Get the scheme of a URI. If the URI does not have a scheme, assume it is a file URI.
|
|
|
|
Parameters
|
|
----------
|
|
uri : str
|
|
The URI to parse.
|
|
|
|
Returns
|
|
-------
|
|
str: The scheme of the URI.
|
|
"""
|
|
parsed = urlparse(uri)
|
|
scheme = parsed.scheme
|
|
if not scheme:
|
|
scheme = "file"
|
|
elif scheme in ["s3a", "s3n"]:
|
|
scheme = "s3"
|
|
elif len(scheme) == 1:
|
|
# Windows drive names are parsed as the scheme
|
|
# e.g. "c:\path" -> ParseResult(scheme="c", netloc="", path="/path", ...)
|
|
# So we add special handling here for schemes that are a single character
|
|
scheme = "file"
|
|
return scheme
|
|
|
|
|
|
def get_uri_location(uri: str) -> str:
|
|
"""
|
|
Get the location of a URI. If the parameter is not a url, assumes it is just a path
|
|
|
|
Parameters
|
|
----------
|
|
uri : str
|
|
The URI to parse.
|
|
|
|
Returns
|
|
-------
|
|
str: Location part of the URL, without scheme
|
|
"""
|
|
parsed = urlparse(uri)
|
|
if len(parsed.scheme) == 1:
|
|
# Windows drive names are parsed as the scheme
|
|
# e.g. "c:\path" -> ParseResult(scheme="c", netloc="", path="/path", ...)
|
|
# So we add special handling here for schemes that are a single character
|
|
return uri
|
|
|
|
if not parsed.netloc:
|
|
return parsed.path
|
|
else:
|
|
return parsed.netloc + parsed.path
|
|
|
|
|
|
def fs_from_uri(uri: str) -> Tuple[pa_fs.FileSystem, str]:
|
|
"""
|
|
Get a PyArrow FileSystem from a URI, handling extra environment variables.
|
|
"""
|
|
if get_uri_scheme(uri) == "s3":
|
|
fs = pa_fs.S3FileSystem(
|
|
endpoint_override=os.environ.get("AWS_ENDPOINT"),
|
|
request_timeout=30,
|
|
connect_timeout=30,
|
|
)
|
|
path = get_uri_location(uri)
|
|
return fs, path
|
|
|
|
return pa_fs.FileSystem.from_uri(uri)
|
|
|
|
|
|
def join_uri(base: Union[str, pathlib.Path], *parts: str) -> str:
|
|
"""
|
|
Join a URI with multiple parts, handles both local and remote paths
|
|
|
|
Parameters
|
|
----------
|
|
base : str
|
|
The base URI
|
|
parts : str
|
|
The parts to join to the base URI, each separated by the
|
|
appropriate path separator for the URI scheme and OS
|
|
"""
|
|
if isinstance(base, pathlib.Path):
|
|
return base.joinpath(*parts)
|
|
base = str(base)
|
|
if get_uri_scheme(base) == "file":
|
|
# using pathlib for local paths make this windows compatible
|
|
# `get_uri_scheme` returns `file` for windows drive names (e.g. `c:\path`)
|
|
return str(pathlib.Path(base, *parts))
|
|
# for remote paths, just use os.path.join
|
|
return "/".join([p.rstrip("/") for p in [base, *parts]])
|
|
|
|
|
|
def attempt_import_or_raise(module: str, mitigation=None):
|
|
"""
|
|
Import the specified module. If the module is not installed,
|
|
raise an ImportError with a helpful message.
|
|
|
|
Parameters
|
|
----------
|
|
module : str
|
|
The name of the module to import
|
|
mitigation : Optional[str]
|
|
The package(s) to install to mitigate the error.
|
|
If not provided then the module name will be used.
|
|
"""
|
|
try:
|
|
return importlib.import_module(module)
|
|
except ImportError:
|
|
raise ImportError(f"Please install {mitigation or module}")
|
|
|
|
|
|
def safe_import_pandas():
|
|
try:
|
|
import pandas as pd
|
|
|
|
return pd
|
|
except ImportError:
|
|
return None
|
|
|
|
|
|
def safe_import_polars():
|
|
try:
|
|
import polars as pl
|
|
|
|
return pl
|
|
except ImportError:
|
|
return None
|
|
|
|
|
|
def inf_vector_column_query(schema: pa.Schema) -> str:
|
|
"""
|
|
Get the vector column name
|
|
|
|
Parameters
|
|
----------
|
|
schema : pa.Schema
|
|
The schema of the vector column.
|
|
|
|
Returns
|
|
-------
|
|
str: the vector column name.
|
|
"""
|
|
vector_col_name = ""
|
|
vector_col_count = 0
|
|
for field_name in schema.names:
|
|
field = schema.field(field_name)
|
|
if pa.types.is_fixed_size_list(field.type) and pa.types.is_floating(
|
|
field.type.value_type
|
|
):
|
|
vector_col_count += 1
|
|
if vector_col_count > 1:
|
|
raise ValueError(
|
|
"Schema has more than one vector column. "
|
|
"Please specify the vector column name "
|
|
"for vector search"
|
|
)
|
|
break
|
|
elif vector_col_count == 1:
|
|
vector_col_name = field_name
|
|
if vector_col_count == 0:
|
|
raise ValueError(
|
|
"There is no vector column in the data. "
|
|
"Please specify the vector column name for vector search"
|
|
)
|
|
return vector_col_name
|
|
|
|
|
|
@singledispatch
|
|
def value_to_sql(value):
|
|
raise NotImplementedError("SQL conversion is not implemented for this type")
|
|
|
|
|
|
@value_to_sql.register(str)
|
|
def _(value: str):
|
|
return f"'{value}'"
|
|
|
|
|
|
@value_to_sql.register(int)
|
|
def _(value: int):
|
|
return str(value)
|
|
|
|
|
|
@value_to_sql.register(float)
|
|
def _(value: float):
|
|
return str(value)
|
|
|
|
|
|
@value_to_sql.register(bool)
|
|
def _(value: bool):
|
|
return str(value).upper()
|
|
|
|
|
|
@value_to_sql.register(type(None))
|
|
def _(value: type(None)):
|
|
return "NULL"
|
|
|
|
|
|
@value_to_sql.register(datetime)
|
|
def _(value: datetime):
|
|
return f"'{value.isoformat()}'"
|
|
|
|
|
|
@value_to_sql.register(date)
|
|
def _(value: date):
|
|
return f"'{value.isoformat()}'"
|
|
|
|
|
|
@value_to_sql.register(list)
|
|
def _(value: list):
|
|
return "[" + ", ".join(map(value_to_sql, value)) + "]"
|
|
|
|
|
|
@value_to_sql.register(np.ndarray)
|
|
def _(value: np.ndarray):
|
|
return value_to_sql(value.tolist())
|