use bson::oid::ObjectId;
use indexmap::IndexMap;
use vantage_core::{Result, error};
use vantage_table::column::core::Column as TableColumn;
use vantage_table::column::flags::ColumnFlag;
use vantage_table::table::Table;
use vantage_table::traits::column_like::ColumnLike;
use vantage_types::{EmptyEntity, Entity};
use vantage_vista::{
Column as VistaColumn, NoExtras, Vista, VistaCapabilities, VistaFactory, VistaMetadata,
flags as vista_flags,
};
use crate::mongodb::MongoDB;
use crate::types::AnyMongoType;
use crate::vista::source::MongoVistaSource;
use crate::vista::spec::{MongoColumnExtras, MongoTableExtras, MongoVistaSpec};
pub struct MongoVistaFactory {
pub(crate) mongo: MongoDB,
}
impl MongoVistaFactory {
pub fn new(mongo: MongoDB) -> Self {
Self { mongo }
}
pub fn from_table<E>(&self, table: Table<MongoDB, E>) -> Result<Vista>
where
E: Entity<AnyMongoType> + 'static,
{
let name = table.table_name().to_string();
let any_table = table.into_entity::<EmptyEntity>();
let column_paths = paths_from_table_columns(&any_table);
Ok(self.wrap_with_paths(any_table, column_paths, name))
}
fn wrap_with_paths(
&self,
table: Table<MongoDB, EmptyEntity>,
column_paths: IndexMap<String, Vec<String>>,
name: String,
) -> Vista {
let metadata = metadata_from_table(&table);
let source = MongoVistaSource::new(
table,
VistaCapabilities {
can_count: true,
can_insert: true,
can_update: true,
can_delete: true,
..VistaCapabilities::default()
},
column_paths,
);
Vista::new(name, Box::new(source), metadata)
}
pub(crate) fn paths_from_spec(
&self,
spec: &MongoVistaSpec,
) -> Result<IndexMap<String, Vec<String>>> {
let mut paths = IndexMap::new();
for (name, col_spec) in &spec.columns {
let path = match col_spec.driver.mongo.as_ref() {
Some(block) => block
.resolved_path(name)?
.unwrap_or_else(|| vec![name.clone()]),
None => vec![name.clone()],
};
paths.insert(name.clone(), path);
}
Ok(paths)
}
pub fn table_from_spec(&self, spec: &MongoVistaSpec) -> Result<Table<MongoDB, EmptyEntity>> {
let collection = spec
.driver
.mongo
.as_ref()
.and_then(|m| m.collection.clone())
.unwrap_or_else(|| spec.name.clone());
let mut table = Table::<MongoDB, EmptyEntity>::new(collection, self.mongo.clone());
for (name, col_spec) in &spec.columns {
table.add_column(build_column(name, col_spec)?);
if col_spec.flags.iter().any(|f| f == vista_flags::TITLE) {
table.add_title_field(name);
}
}
let id_column = resolve_id_column(spec);
if !table.columns().contains_key(&id_column) {
return Err(error!(
"id column not present in spec.columns",
id = id_column
));
}
table.set_id_field(&id_column);
Ok(table)
}
}
pub(crate) fn resolve_id_column(spec: &MongoVistaSpec) -> String {
if let Some(id) = &spec.id_column {
return id.clone();
}
for (name, col_spec) in &spec.columns {
if col_spec.flags.iter().any(|f| f == vista_flags::ID) {
return name.clone();
}
}
"_id".to_string()
}
pub(crate) fn build_column(
name: &str,
col_spec: &vantage_vista::ColumnSpec<MongoColumnExtras>,
) -> Result<TableColumn<AnyMongoType>> {
let ty = col_spec.col_type.as_deref().unwrap_or("string");
let hidden = col_spec.flags.iter().any(|f| f == vista_flags::HIDDEN);
let mut col = column_for_type(name, ty)?;
if hidden {
col = col.with_flag(ColumnFlag::Hidden);
}
Ok(col)
}
pub(crate) fn column_for_type(name: &str, ty: &str) -> Result<TableColumn<AnyMongoType>> {
let col: TableColumn<AnyMongoType> = match ty {
"int" | "integer" | "i64" | "i32" => {
TableColumn::from_column(TableColumn::<i64>::new(name))
}
"float" | "double" | "f64" => TableColumn::from_column(TableColumn::<f64>::new(name)),
"bool" | "boolean" => TableColumn::from_column(TableColumn::<bool>::new(name)),
"string" | "text" | "str" => TableColumn::from_column(TableColumn::<String>::new(name)),
"object_id" | "objectid" | "oid" => {
TableColumn::from_column(TableColumn::<ObjectId>::new(name))
}
other => {
return Err(error!(
"Unknown YAML column type",
column = name,
ty = other.to_string()
));
}
};
Ok(col)
}
pub(crate) fn paths_from_table_columns<T, E>(table: &Table<T, E>) -> IndexMap<String, Vec<String>>
where
T: vantage_table::traits::table_source::TableSource,
E: Entity<T::Value>,
T::Column<T::AnyType>: ColumnLike<T::AnyType>,
{
let mut paths = IndexMap::new();
for (name, col) in table.columns() {
let path = match col.alias() {
Some(a) => vec![a.to_string()],
None => vec![name.clone()],
};
paths.insert(name.clone(), path);
}
paths
}
pub(crate) fn metadata_from_table<T, E>(table: &Table<T, E>) -> VistaMetadata
where
T: vantage_table::traits::table_source::TableSource,
E: Entity<T::Value>,
T::Column<T::AnyType>: ColumnLike<T::AnyType>,
{
let mut metadata = VistaMetadata::new();
for (name, col) in table.columns() {
let mut vc = VistaColumn::new(name.clone(), col.get_type().to_string());
if col.flags().contains(&ColumnFlag::Hidden) {
vc = vc.with_flag(vista_flags::HIDDEN);
}
metadata = metadata.with_column(vc);
}
if let Some(id_field) = table.id_field() {
metadata = metadata.with_id_column(id_field.name().to_string());
}
for title in table.title_fields() {
if let Some(col) = metadata.columns.get_mut(title) {
col.flags.push(vista_flags::TITLE.to_string());
}
}
metadata
}
impl VistaFactory for MongoVistaFactory {
type TableExtras = MongoTableExtras;
type ColumnExtras = MongoColumnExtras;
type ReferenceExtras = NoExtras;
fn build_from_spec(&self, spec: MongoVistaSpec) -> Result<Vista> {
let column_paths = self.paths_from_spec(&spec)?;
let table = self.table_from_spec(&spec)?;
Ok(self.wrap_with_paths(table, column_paths, spec.name))
}
}