use std::sync::{Arc, Weak}; use collab_database::rows::RowId; use lib_infra::box_any::BoxAny; use tokio::sync::oneshot; use tracing::error; use flowy_error::{FlowyError, FlowyResult}; use lib_dispatch::prelude::{af_spawn, data_result_ok, AFPluginData, AFPluginState, DataResult}; use crate::entities::*; use crate::manager::DatabaseManager; use crate::services::field::{ type_option_data_from_pb, ChecklistCellChangeset, DateCellChangeset, RelationCellChangeset, SelectOptionCellChangeset, }; use crate::services::field_settings::FieldSettingsChangesetParams; use crate::services::group::GroupChangeset; use crate::services::share::csv::CSVFormat; fn upgrade_manager( database_manager: AFPluginState>, ) -> FlowyResult> { let manager = database_manager .upgrade() .ok_or(FlowyError::internal().with_context("The database manager is already dropped"))?; Ok(manager) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn get_database_data_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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_all, err)] pub(crate) async fn open_database_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let manager = upgrade_manager(manager)?; let view_id: DatabaseViewIdPB = data.into_inner(); let database_id = manager .get_database_id_with_view_id(view_id.as_ref()) .await?; let _ = manager.open_database(&database_id).await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn get_database_id_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let view_id: DatabaseViewIdPB = data.into_inner(); let database_id = manager .get_database_id_with_view_id(view_id.as_ref()) .await?; data_result_ok(DatabaseIdPB { value: database_id }) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_database_setting_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; let params = data.try_into_inner()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; if let Some(payload) = params.insert_filter { database_editor .modify_view_filters(¶ms.view_id, payload.try_into()?) .await?; } if let Some(payload) = params.update_filter_type { database_editor .modify_view_filters(¶ms.view_id, payload.try_into()?) .await?; } if let Some(payload) = params.update_filter_data { database_editor .modify_view_filters(¶ms.view_id, payload.try_into()?) .await?; } if let Some(payload) = params.delete_filter { database_editor .modify_view_filters(¶ms.view_id, payload.into()) .await?; } if let Some(update_sort) = params.update_sort { let _ = database_editor.create_or_update_sort(update_sort).await?; } if let Some(reorder_sort) = params.reorder_sort { database_editor.reorder_sort(reorder_sort).await?; } if let Some(delete_sort) = params.delete_sort { database_editor.delete_sort(delete_sort).await?; } if let Some(layout_type) = params.layout_type { database_editor .update_view_layout(¶ms.view_id, layout_type.into()) .await?; } Ok(()) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_all_filters_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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::new) .collect::>() .into(); data_result_ok(fields) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn get_primary_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let view_id = data.into_inner().value; let database_editor = manager.get_database_with_view_id(&view_id).await?; let mut fields = database_editor .get_fields(&view_id, None) .into_iter() .filter(|field| field.is_primary) .map(FieldPB::new) .collect::>(); if fields.is_empty() { // The primary field should not be empty. Because it is created when the database is created. // If it is empty, it must be a bug. Err(FlowyError::record_not_found()) } else { if fields.len() > 1 { // The primary field should not be more than one. If it is more than one, // it must be a bug. tracing::error!("The primary field is more than one"); } data_result_ok(fields.remove(0)) } } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn update_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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(params.type_option_data, &field_type)?; database_editor .update_field_type_option(¶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 manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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, params.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.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 manager = upgrade_manager(manager)?; let params: DuplicateFieldPayloadPB = data.into_inner(); 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(()) } /// Create a field and save it. Returns the [FieldPB] in the current view. #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn create_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let params: CreateFieldParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let data = database_editor .create_field_with_type_option(params) .await?; 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 manager = upgrade_manager(manager)?; 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(params).await?; Ok(()) } // #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn get_row_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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.view_id, ¶ms.row_id) .map(RowPB::from); data_result_ok(OptionalRowPB { row }) } pub(crate) async fn get_row_meta_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let params: RowIdParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; match database_editor.get_row_meta(¶ms.view_id, ¶ms.row_id) { None => Err(FlowyError::record_not_found()), Some(row) => data_result_ok(row), } } pub(crate) async fn update_row_meta_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; let params: UpdateRowMetaParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let row_id = RowId::from(params.id.clone()); database_editor.update_row_meta(&row_id, params).await; Ok(()) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn delete_row_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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, ¶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 manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; let params = data.try_into_inner()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; match database_editor.create_row(params).await? { Some(row) => data_result_ok(RowMetaPB::from(row)), None => Err(FlowyError::internal().with_context("Error creating row")), } } // #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn get_cell_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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_pb(¶ms.field_id, ¶ms.row_id) .await .unwrap_or_else(|| CellPB::empty(¶ms.field_id, params.row_id.into_inner())); 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 manager = upgrade_manager(manager)?; 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, BoxAny::new(params.cell_changeset), ) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn new_select_option_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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() .with_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 manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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, BoxAny::new(changeset), ) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn update_checklist_cell_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let manager = upgrade_manager(manager)?; let params: ChecklistCellDataChangesetParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; let changeset = ChecklistCellChangeset { insert_options: params .insert_options .into_iter() .map(|name| (name, false)) .collect(), selected_option_ids: params.selected_option_ids, delete_option_ids: params.delete_option_ids, update_options: params.update_options, }; database_editor .update_cell_with_changeset( ¶ms.view_id, params.row_id, ¶ms.field_id, BoxAny::new(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 manager = upgrade_manager(manager)?; let data = data.into_inner(); let cell_id: CellIdParams = data.cell_id.try_into()?; let cell_changeset = DateCellChangeset { date: data.date, time: data.time, end_date: data.end_date, end_time: data.end_time, include_time: data.include_time, is_range: data.is_range, clear_flag: data.clear_flag, reminder_id: data.reminder_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, BoxAny::new(cell_changeset), ) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn get_groups_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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 = "trace", skip_all, err)] pub(crate) async fn set_group_by_field_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; let params: GroupByFieldParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .set_group_by_field(¶ms.view_id, ¶ms.field_id) .await?; Ok(()) } #[tracing::instrument(level = "trace", skip_all, err)] pub(crate) async fn update_group_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; let params: UpdateGroupParams = data.into_inner().try_into()?; let view_id = params.view_id.clone(); let database_editor = manager.get_database_with_view_id(&view_id).await?; let group_changeset = GroupChangeset::from(params); let (tx, rx) = oneshot::channel(); af_spawn(async move { let result = database_editor .update_group(&view_id, vec![group_changeset].into()) .await; let _ = tx.send(result); }); let _ = rx.await?; Ok(()) } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn move_group_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; 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 manager = upgrade_manager(manager)?; 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.from_group_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 create_group_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; let params: CreateGroupParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .create_group(¶ms.view_id, ¶ms.name) .await?; Ok(()) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn delete_group_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; let params: DeleteGroupParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor.delete_group(params).await?; Ok(()) } #[tracing::instrument(level = "debug", skip(manager), err)] pub(crate) async fn get_databases_handler( manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let metas = manager.get_all_databases_meta().await; let mut items = Vec::with_capacity(metas.len()); for meta in metas { match manager.get_database_inline_view_id(&meta.database_id).await { Ok(view_id) => items.push(DatabaseMetaPB { database_id: meta.database_id, inline_view_id: view_id, }), Err(err) => { error!(?err); }, } } let data = RepeatedDatabaseDescriptionPB { items }; 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 manager = upgrade_manager(manager)?; let changeset = data.into_inner(); let view_id = changeset.view_id.clone(); let params: LayoutSettingChangeset = changeset.try_into()?; let database_editor = manager.get_database_with_view_id(&view_id).await?; database_editor.set_layout_setting(&view_id, params).await?; Ok(()) } pub(crate) async fn get_layout_setting_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let params: DatabaseLayoutMeta = 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(DatabaseLayoutSettingPB::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 manager = upgrade_manager(manager)?; 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_no_date_calendar_events_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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_no_date_calendar_events(¶ms.view_id) .await; todo!() } #[tracing::instrument(level = "debug", skip(data, manager), err)] pub(crate) async fn get_calendar_event_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; 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 move_calendar_event_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; let data = data.into_inner(); let cell_id: CellIdParams = data.cell_path.try_into()?; let cell_changeset = DateCellChangeset { date: Some(data.timestamp), ..Default::default() }; 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, BoxAny::new(cell_changeset), ) .await?; Ok(()) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn create_database_view( _data: AFPluginData, _manager: AFPluginState>, ) -> FlowyResult<()> { // let data: CreateDatabaseViewParams = data.into_inner().try_into()?; Ok(()) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn export_csv_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let view_id = data.into_inner().value; let database = manager.get_database_with_view_id(&view_id).await?; let data = database.export_csv(CSVFormat::Original).await?; data_result_ok(DatabaseExportDataPB { export_type: DatabaseExportDataType::CSV, data, }) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn get_snapshots_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let view_id = data.into_inner().value; let snapshots = manager.get_database_snapshots(&view_id, 10).await?; data_result_ok(RepeatedDatabaseSnapshotPB { items: snapshots }) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn get_field_settings_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let (view_id, field_ids) = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(&view_id).await?; let field_settings = database_editor .get_field_settings(&view_id, field_ids.clone()) .await? .into_iter() .map(FieldSettingsPB::from) .collect(); data_result_ok(RepeatedFieldSettingsPB { items: field_settings, }) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn get_all_field_settings_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let view_id = data.into_inner(); let database_editor = manager.get_database_with_view_id(view_id.as_ref()).await?; let field_settings = database_editor .get_all_field_settings(view_id.as_ref()) .await? .into_iter() .map(FieldSettingsPB::from) .collect(); data_result_ok(RepeatedFieldSettingsPB { items: field_settings, }) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn update_field_settings_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; let params: FieldSettingsChangesetParams = data.into_inner().try_into()?; let database_editor = manager.get_database_with_view_id(¶ms.view_id).await?; database_editor .update_field_settings_with_changeset(params) .await?; Ok(()) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn get_all_calculations_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let view_id = data.into_inner(); let database_editor = manager.get_database_with_view_id(view_id.as_ref()).await?; let calculations = database_editor.get_all_calculations(view_id.as_ref()).await; data_result_ok(calculations) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn update_calculation_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let manager = upgrade_manager(manager)?; let params: UpdateCalculationChangesetPB = data.into_inner(); let editor = manager.get_database_with_view_id(¶ms.view_id).await?; editor.update_calculation(params).await?; Ok(()) } #[tracing::instrument(level = "trace", skip(data, manager), err)] pub(crate) async fn remove_calculation_handler( data: AFPluginData, manager: AFPluginState>, ) -> Result<(), FlowyError> { let manager = upgrade_manager(manager)?; let params: RemoveCalculationChangesetPB = data.into_inner(); let editor = manager.get_database_with_view_id(¶ms.view_id).await?; editor.remove_calculation(params).await?; Ok(()) } pub(crate) async fn get_related_database_ids_handler( _data: AFPluginData, _manager: AFPluginState>, ) -> FlowyResult<()> { Ok(()) } #[tracing::instrument(level = "debug", skip_all, err)] pub(crate) async fn update_relation_cell_handler( data: AFPluginData, manager: AFPluginState>, ) -> FlowyResult<()> { let manager = upgrade_manager(manager)?; let params: RelationCellChangesetPB = data.into_inner(); let view_id = parser::NotEmptyStr::parse(params.view_id) .map_err(|_| flowy_error::ErrorCode::DatabaseViewIdIsEmpty)? .0; let cell_id: CellIdParams = params.cell_id.try_into()?; let params = RelationCellChangeset { inserted_row_ids: params .inserted_row_ids .into_iter() .map(Into::into) .collect(), removed_row_ids: params.removed_row_ids.into_iter().map(Into::into).collect(), }; let database_editor = manager.get_database_with_view_id(&view_id).await?; // // get the related database // let related_database_id = database_editor // .get_related_database_id(&cell_id.field_id) // .await?; // let related_database_editor = manager.get_database(&related_database_id).await?; // // validate the changeset contents // related_database_editor // .validate_row_ids_exist(¶ms) // .await?; // update the cell in the database database_editor .update_cell_with_changeset( &view_id, cell_id.row_id, &cell_id.field_id, BoxAny::new(params), ) .await?; Ok(()) } pub(crate) async fn get_related_row_datas_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let params: RepeatedRowIdPB = data.into_inner(); let database_editor = manager.get_database(¶ms.database_id).await?; let row_datas = database_editor .get_related_rows(Some(¶ms.row_ids)) .await?; data_result_ok(RepeatedRelatedRowDataPB { rows: row_datas }) } pub(crate) async fn get_related_database_rows_handler( data: AFPluginData, manager: AFPluginState>, ) -> DataResult { let manager = upgrade_manager(manager)?; let database_id = data.into_inner().value; let database_editor = manager.get_database(&database_id).await?; let row_datas = database_editor.get_related_rows(None).await?; data_result_ok(RepeatedRelatedRowDataPB { rows: row_datas }) }