use collab_database::database::gen_row_id; use std::sync::Arc; use collab_database::rows::RowId; use collab_database::views::DatabaseLayout; use lib_infra::util::timestamp; use flowy_error::{ErrorCode, FlowyError, FlowyResult}; use lib_dispatch::prelude::{data_result_ok, AFPluginData, AFPluginState, DataResult}; use crate::entities::*; use crate::manager::DatabaseManager2; use crate::services::cell::CellBuilder; use crate::services::field::{ type_option_data_from_pb_or_default, DateCellChangeset, SelectOptionCellChangeset, }; use crate::services::share::csv::CSVFormat; #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_database_data_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let view_id: DatabaseViewIdPB = data.into_inner(); let database_editor = manager.get_database_with_view_id(view_id.as_ref()).await?; let data = database_editor.get_database_data(view_id.as_ref()).await; data_result_ok(data) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_database_setting_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let view_id: DatabaseViewIdPB = data.into_inner(); let database_editor = manager.get_database_with_view_id(view_id.as_ref()).await?; let data = database_editor .get_database_view_setting(view_id.as_ref()) .await?; data_result_ok(data) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn update_database_setting_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: DatabaseSettingChangesetParams = data.into_inner().try_into()?; let editor = manager.get_database_with_view_id(¶ms.view_id).await?; if let Some(insert_params) = params.insert_group { editor.insert_group(insert_params).await?; } if let Some(delete_params) = params.delete_group { editor.delete_group(delete_params).await?; } if let Some(alter_filter) = params.insert_filter { editor.create_or_update_filter(alter_filter).await?; } if let Some(delete_filter) = params.delete_filter { editor.delete_filter(delete_filter).await?; } if let Some(alter_sort) = params.alert_sort { let _ = editor.create_or_update_sort(alter_sort).await?; } if let Some(delete_sort) = params.delete_sort { editor.delete_sort(delete_sort).await?; } Ok(()) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_all_filters_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let view_id: DatabaseViewIdPB = data.into_inner(); let database_editor = manager.get_database_with_view_id(view_id.as_ref()).await?; let filters = database_editor.get_all_filters(view_id.as_ref()).await; data_result_ok(filters) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_all_sorts_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let view_id: DatabaseViewIdPB = data.into_inner(); let database_editor = manager.get_database_with_view_id(view_id.as_ref()).await?; let sorts = database_editor.get_all_sorts(view_id.as_ref()).await; data_result_ok(sorts) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn delete_all_sorts_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let view_id: DatabaseViewIdPB = data.into_inner(); let database_editor = manager.get_database_with_view_id(view_id.as_ref()).await?; database_editor.delete_all_sorts(view_id.as_ref()).await; Ok(()) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_fields_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: GetFieldParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let fields = database_editor .get_fields(¶ms.view_id, params.field_ids) .into_iter() .map(FieldPB::from) .collect::>() .into(); data_result_ok(fields) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn update_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: FieldChangesetParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor.update_field(params).await?; Ok(()) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn update_field_type_option_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: TypeOptionChangesetParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; if let Some(old_field) = database_editor.get_field(¶ms.field_id) { let field_type = FieldType::from(old_field.field_type); let type_option_data = type_option_data_from_pb_or_default(params.type_option_data, &field_type); database_editor .update_field_type_option( ¶ms.view_id, ¶ms.field_id, type_option_data, old_field, ) .await?; } Ok(()) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn delete_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: FieldIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor.delete_field(¶ms.field_id).await?; Ok(()) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn switch_to_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: EditFieldParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let old_field = database_editor.get_field(¶ms.field_id); database_editor .switch_to_field_type(¶ms.field_id, ¶ms.field_type) .await?; if let Some(new_type_option) = database_editor .get_field(¶ms.field_id) .map(|field| field.get_any_type_option(field.field_type)) { match (old_field, new_type_option) { (Some(old_field), Some(new_type_option)) => { database_editor .update_field_type_option( ¶ms.view_id, ¶ms.field_id, new_type_option, old_field, ) .await?; }, _ => { tracing::warn!("Old field and the new type option should not be empty"); }, } } Ok(()) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn duplicate_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: FieldIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .duplicate_field(¶ms.view_id, ¶ms.field_id) .await?; Ok(()) } /// Return the FieldTypeOptionData if the Field exists otherwise return record not found error. #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_field_type_option_data_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: TypeOptionPathParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; if let Some((field, data)) = database_editor .get_field_type_option_data(¶ms.field_id) .await { let data = TypeOptionPB { view_id: params.view_id, field: FieldPB::from(field), type_option_data: data.to_vec(), }; data_result_ok(data) } else { Err(FlowyError::record_not_found()) } } /// Create FieldMeta and save it. Return the FieldTypeOptionData. #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn create_field_type_option_data_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: CreateFieldParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let (field, data) = database_editor .create_field_with_type_option(¶ms.view_id, ¶ms.field_type, params.type_option_data) .await; let data = TypeOptionPB { view_id: params.view_id, field: FieldPB::from(field), type_option_data: data.to_vec(), }; data_result_ok(data) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn move_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: MoveFieldParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .move_field( ¶ms.view_id, ¶ms.field_id, params.from_index, params.to_index, ) .await?; Ok(()) } // #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn get_row_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: RowIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let row = database_editor.get_row(¶ms.row_id).map(RowPB::from); data_result_ok(OptionalRowPB { row }) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn delete_row_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: RowIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor.delete_row(¶ms.row_id).await; Ok(()) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn duplicate_row_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: RowIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .duplicate_row(¶ms.view_id, params.group_id, ¶ms.row_id) .await; Ok(()) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn move_row_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: MoveRowParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .move_row(¶ms.view_id, params.from_row_id, params.to_row_id) .await; Ok(()) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn create_row_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: CreateRowParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let fields = database_editor.get_fields(¶ms.view_id, None); let cells = CellBuilder::with_cells(params.cell_data_by_field_id.unwrap_or_default(), &fields).build(); let view_id = params.view_id; let group_id = params.group_id; let params = collab_database::rows::CreateRowParams { id: gen_row_id(), cells, height: 60, visibility: true, prev_row_id: params.start_row_id, timestamp: timestamp(), }; match database_editor .create_row(&view_id, group_id, params) .await? { None => Err(FlowyError::internal().context("Create row fail")), Some(row) => data_result_ok(RowPB::from(row)), } } // #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn get_cell_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: CellIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let cell = database_editor .get_cell(¶ms.field_id, params.row_id) .await; data_result_ok(cell) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn update_cell_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: CellChangesetPB = data.into_inner(); let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .update_cell_with_changeset( ¶ms.view_id, RowId::from(params.row_id), ¶ms.field_id, params.cell_changeset.clone(), ) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn new_select_option_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: CreateSelectOptionParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let result = database_editor .create_select_option(¶ms.field_id, params.option_name) .await; match result { None => { Err(FlowyError::record_not_found().context("Create select option fail. Can't find the field")) }, Some(pb) => data_result_ok(pb), } } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn insert_or_update_select_option_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params = data.into_inner(); let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .insert_select_options( ¶ms.view_id, ¶ms.field_id, RowId::from(params.row_id), params.items, ) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn delete_select_option_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params = data.into_inner(); let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .delete_select_options( ¶ms.view_id, ¶ms.field_id, RowId::from(params.row_id), params.items, ) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_select_option_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: CellIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let options = database_editor .get_select_options(params.row_id, ¶ms.field_id) .await; data_result_ok(options) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn update_select_option_cell_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let params: SelectOptionCellChangesetParams = data.into_inner().try_into()?; let database_editor = manager .get_database_with_view_id(¶ms.cell_identifier.view_id) .await?; let changeset = SelectOptionCellChangeset { insert_option_ids: params.insert_option_ids, delete_option_ids: params.delete_option_ids, }; database_editor .update_cell_with_changeset( ¶ms.cell_identifier.view_id, params.cell_identifier.row_id, ¶ms.cell_identifier.field_id, changeset, ) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn update_date_cell_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let data = data.into_inner(); let cell_id: CellIdParams = data.cell_path.try_into()?; let cell_changeset = DateCellChangeset { date: data.date, time: data.time, include_time: data.include_time, timezone_id: data.timezone_id, }; let database_editor = manager.get_database_with_view_id(&cell_id.view_id).await?; database_editor .update_cell_with_changeset( &cell_id.view_id, cell_id.row_id, &cell_id.field_id, cell_changeset, ) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn get_groups_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: DatabaseViewIdPB = data.into_inner(); let database_editor = manager.get_database_with_view_id(params.as_ref()).await?; let groups = database_editor.load_groups(params.as_ref()).await?; data_result_ok(groups) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn get_group_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: DatabaseGroupIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let group = database_editor .get_group(¶ms.view_id, ¶ms.group_id) .await?; data_result_ok(group) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn move_group_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let params: MoveGroupParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .move_group(¶ms.view_id, ¶ms.from_group_id, ¶ms.to_group_id) .await?; Ok(()) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn move_group_row_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let params: MoveGroupRowParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .move_group_row( ¶ms.view_id, ¶ms.to_group_id, params.from_row_id, params.to_row_id, ) .await?; Ok(()) } #[tracing::instrument(level = "debug", skip(manager), err)] pub(crate) async fn get_databases_handler( manager: AFPluginState>, ) -> DataResult { let data = manager.get_all_databases_description().await; data_result_ok(data) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn set_layout_setting_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let params: LayoutSettingChangeset = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let layout_params = LayoutSettingParams { calendar: params.calendar, }; database_editor .set_layout_setting(¶ms.view_id, DatabaseLayout::Calendar, layout_params) .await; Ok(()) } pub(crate) async fn get_layout_setting_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: DatabaseLayoutId = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let layout_setting_pb = database_editor .get_layout_setting(¶ms.view_id, params.layout) .await .map(LayoutSettingPB::from) .unwrap_or_default(); data_result_ok(layout_setting_pb) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn get_calendar_events_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: CalendarEventRequestParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let events = database_editor .get_all_calendar_events(¶ms.view_id) .await; data_result_ok(RepeatedCalendarEventPB { items: events }) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn get_calendar_event_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let params: RowIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let event = database_editor .get_calendar_event(¶ms.view_id, params.row_id) .await; match event { None => Err(FlowyError::record_not_found()), Some(event) => data_result_ok(event), } } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn import_data_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let params = data.into_inner(); match params.import_type { ImportTypePB::CSV => { if let Some(data) = params.data { manager.import_csv(data, CSVFormat::META).await?; } else if let Some(uri) = params.uri { manager.import_csv_from_uri(uri, CSVFormat::META).await?; } else { return Err(FlowyError::new( ErrorCode::InvalidData, "No data or uri provided", )); } }, } Ok(()) }