use http_req::{
request::{Method, Request},
uri::Uri,
};
use lazy_static::lazy_static;
use serde::{Deserialize, Serialize};
use serde_json::{Map, Value};
lazy_static! {
static ref VECTOR_STORE_API_PREFIX: String = String::from(
std::option_env!("VECTOR_STORE_API_PREFIX")
.unwrap_or("https://vector-store.flows.network/api")
);
}
extern "C" {
fn get_flows_user(p: *mut u8) -> i32;
fn get_flow_id(p: *mut u8) -> i32;
}
unsafe fn _get_flows_user() -> String {
let mut flows_user = Vec::<u8>::with_capacity(100);
let c = get_flows_user(flows_user.as_mut_ptr());
flows_user.set_len(c as usize);
String::from_utf8(flows_user).unwrap()
}
unsafe fn _get_flow_id() -> String {
let mut flow_id = Vec::<u8>::with_capacity(100);
let c = get_flow_id(flow_id.as_mut_ptr());
if c == 0 {
panic!("Failed to get flow id");
}
flow_id.set_len(c as usize);
String::from_utf8(flow_id).unwrap()
}
#[derive(Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct CollectionInfo {
pub points_count: u64,
}
#[derive(Debug, Serialize)]
#[serde(rename_all = "camelCase")]
pub struct CollectionCreateParams {
pub vector_size: u64,
}
#[derive(Debug, Serialize, Deserialize)]
#[serde(untagged)]
pub enum PointId {
Uuid(String),
Num(u64),
}
#[derive(Debug, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct Point {
pub id: PointId,
pub vector: Vec<f32>,
pub payload: Option<Map<String, Value>>,
}
#[derive(Debug, Serialize)]
#[serde(rename_all = "camelCase")]
pub struct PointsSearchParams {
pub vector: Vec<f32>,
pub limit: u64,
}
#[derive(Debug, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct ScoredPoint {
pub id: PointId,
pub vector: Option<Vec<f32>>,
pub payload: Option<Map<String, Value>>,
pub score: f32,
}
pub async fn collection_info(collection_name: &str) -> Result<CollectionInfo, String> {
unsafe {
let flows_user = _get_flows_user();
let flow_id = _get_flow_id();
let mut writer = Vec::new();
let uri = format!(
"{}/{}/{}/{}/collectionInfo",
VECTOR_STORE_API_PREFIX.as_str(),
flows_user,
flow_id,
collection_name,
);
let uri = Uri::try_from(uri.as_str()).unwrap();
match Request::new(&uri)
.method(Method::GET)
.header("Content-Type", "application/json")
.send(&mut writer)
{
Ok(res) => {
if res.status_code().is_success() {
serde_json::from_slice::<CollectionInfo>(&writer)
.or_else(|e| Err(e.to_string()))
} else {
let err = String::from_utf8_lossy(&writer);
log::error!("{}", err);
Err(err.into_owned())
}
}
Err(e) => Err(e.to_string()),
}
}
}
pub async fn create_collection(
collection_name: &str,
params: &CollectionCreateParams,
) -> Result<(), String> {
unsafe {
let flows_user = _get_flows_user();
let flow_id = _get_flow_id();
let mut writer = Vec::new();
let uri = format!(
"{}/{}/{}/{}/createCollection",
VECTOR_STORE_API_PREFIX.as_str(),
flows_user,
flow_id,
collection_name,
);
let uri = Uri::try_from(uri.as_str()).unwrap();
let body = serde_json::to_vec(¶ms).unwrap_or_default();
match Request::new(&uri)
.method(Method::PUT)
.header("Content-Type", "application/json")
.header("Content-Length", &body.len())
.body(&body)
.send(&mut writer)
{
Ok(res) => {
if res.status_code().is_success() {
Ok(())
} else {
let err = String::from_utf8_lossy(&writer);
log::error!("{}", err);
Err(err.into_owned())
}
}
Err(e) => Err(e.to_string()),
}
}
}
pub async fn delete_collection(collection_name: &str) -> Result<(), String> {
unsafe {
let flows_user = _get_flows_user();
let flow_id = _get_flow_id();
let mut writer = Vec::new();
let uri = format!(
"{}/{}/{}/{}/deleteCollection",
VECTOR_STORE_API_PREFIX.as_str(),
flows_user,
flow_id,
collection_name,
);
let uri = Uri::try_from(uri.as_str()).unwrap();
match Request::new(&uri)
.method(Method::DELETE)
.header("Content-Type", "application/json")
.send(&mut writer)
{
Ok(res) => {
if res.status_code().is_success() {
Ok(())
} else {
let err = String::from_utf8_lossy(&writer);
log::error!("{}", err);
Err(err.into_owned())
}
}
Err(e) => Err(e.to_string()),
}
}
}
pub async fn upsert_points(collection_name: &str, points: Vec<Point>) -> Result<(), String> {
unsafe {
let flows_user = _get_flows_user();
let flow_id = _get_flow_id();
let mut writer = Vec::new();
let uri = format!(
"{}/{}/{}/{}/upsertPoints",
VECTOR_STORE_API_PREFIX.as_str(),
flows_user,
flow_id,
collection_name,
);
let uri = Uri::try_from(uri.as_str()).unwrap();
let body = serde_json::to_vec(&points).unwrap_or_default();
match Request::new(&uri)
.method(Method::PUT)
.header("Content-Type", "application/json")
.header("Content-Length", &body.len())
.body(&body)
.send(&mut writer)
{
Ok(res) => {
if res.status_code().is_success() {
Ok(())
} else {
let err = String::from_utf8_lossy(&writer);
log::error!("{}", err);
Err(err.into_owned())
}
}
Err(e) => Err(e.to_string()),
}
}
}
pub async fn search_points(
collection_name: &str,
params: &PointsSearchParams,
) -> Result<Vec<ScoredPoint>, String> {
unsafe {
let flows_user = _get_flows_user();
let flow_id = _get_flow_id();
let mut writer = Vec::new();
let uri = format!(
"{}/{}/{}/{}/searchPoints",
VECTOR_STORE_API_PREFIX.as_str(),
flows_user,
flow_id,
collection_name,
);
let uri = Uri::try_from(uri.as_str()).unwrap();
let body = serde_json::to_vec(¶ms).unwrap_or_default();
match Request::new(&uri)
.method(Method::POST)
.header("Content-Type", "application/json")
.header("Content-Length", &body.len())
.body(&body)
.send(&mut writer)
{
Ok(res) => {
if res.status_code().is_success() {
serde_json::from_slice::<Vec<ScoredPoint>>(&writer)
.or_else(|e| Err(e.to_string()))
} else {
let err = String::from_utf8_lossy(&writer);
log::error!("{}", err);
Err(err.into_owned())
}
}
Err(e) => Err(e.to_string()),
}
}
}