lib.rs 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. #![allow(clippy::not_unsafe_ptr_arg_deref)]
  2. mod c;
  3. mod model;
  4. mod notification;
  5. mod protobuf;
  6. mod util;
  7. use crate::notification::DartNotificationSender;
  8. use crate::{
  9. c::{extend_front_four_bytes_into_bytes, forget_rust},
  10. model::{FFIRequest, FFIResponse},
  11. };
  12. use flowy_core::get_client_server_configuration;
  13. use flowy_core::*;
  14. use flowy_notification::register_notification_sender;
  15. use lazy_static::lazy_static;
  16. use lib_dispatch::prelude::ToBytes;
  17. use lib_dispatch::prelude::*;
  18. use parking_lot::RwLock;
  19. use std::{ffi::CStr, os::raw::c_char};
  20. lazy_static! {
  21. static ref APPFLOWY_CORE: RwLock<Option<AppFlowyCore>> = RwLock::new(None);
  22. }
  23. #[no_mangle]
  24. pub extern "C" fn init_sdk(path: *mut c_char) -> i64 {
  25. let c_str: &CStr = unsafe { CStr::from_ptr(path) };
  26. let path: &str = c_str.to_str().unwrap();
  27. let server_config = get_client_server_configuration().unwrap();
  28. let log_crates = vec!["flowy-ffi".to_string()];
  29. let config = AppFlowyCoreConfig::new(path, DEFAULT_NAME.to_string(), server_config)
  30. .log_filter("info", log_crates);
  31. *APPFLOWY_CORE.write() = Some(AppFlowyCore::new(config));
  32. 0
  33. }
  34. #[no_mangle]
  35. pub extern "C" fn async_event(port: i64, input: *const u8, len: usize) {
  36. let request: AFPluginRequest = FFIRequest::from_u8_pointer(input, len).into();
  37. log::trace!(
  38. "[FFI]: {} Async Event: {:?} with {} port",
  39. &request.id,
  40. &request.event,
  41. port
  42. );
  43. let dispatcher = match APPFLOWY_CORE.read().as_ref() {
  44. None => {
  45. log::error!("sdk not init yet.");
  46. return;
  47. },
  48. Some(e) => e.event_dispatcher.clone(),
  49. };
  50. let _ = AFPluginDispatcher::async_send_with_callback(
  51. dispatcher,
  52. request,
  53. move |resp: AFPluginEventResponse| {
  54. log::trace!("[FFI]: Post data to dart through {} port", port);
  55. Box::pin(post_to_flutter(resp, port))
  56. },
  57. );
  58. }
  59. #[no_mangle]
  60. pub extern "C" fn sync_event(input: *const u8, len: usize) -> *const u8 {
  61. let request: AFPluginRequest = FFIRequest::from_u8_pointer(input, len).into();
  62. log::trace!("[FFI]: {} Sync Event: {:?}", &request.id, &request.event,);
  63. let dispatcher = match APPFLOWY_CORE.read().as_ref() {
  64. None => {
  65. log::error!("sdk not init yet.");
  66. return forget_rust(Vec::default());
  67. },
  68. Some(e) => e.event_dispatcher.clone(),
  69. };
  70. let _response = AFPluginDispatcher::sync_send(dispatcher, request);
  71. // FFIResponse { }
  72. let response_bytes = vec![];
  73. let result = extend_front_four_bytes_into_bytes(&response_bytes);
  74. forget_rust(result)
  75. }
  76. #[no_mangle]
  77. pub extern "C" fn set_stream_port(port: i64) -> i32 {
  78. register_notification_sender(DartNotificationSender::new(port));
  79. 0
  80. }
  81. #[inline(never)]
  82. #[no_mangle]
  83. pub extern "C" fn link_me_please() {}
  84. #[inline(always)]
  85. async fn post_to_flutter(response: AFPluginEventResponse, port: i64) {
  86. let isolate = allo_isolate::Isolate::new(port);
  87. match isolate
  88. .catch_unwind(async {
  89. let ffi_resp = FFIResponse::from(response);
  90. ffi_resp.into_bytes().unwrap().to_vec()
  91. })
  92. .await
  93. {
  94. Ok(_success) => {
  95. log::trace!("[FFI]: Post data to dart success");
  96. },
  97. Err(e) => {
  98. if let Some(msg) = e.downcast_ref::<&str>() {
  99. log::error!("[FFI]: {:?}", msg);
  100. } else {
  101. log::error!("[FFI]: allo_isolate post panic");
  102. }
  103. },
  104. }
  105. }
  106. #[no_mangle]
  107. pub extern "C" fn backend_log(level: i64, data: *const c_char) {
  108. let c_str = unsafe { CStr::from_ptr(data) };
  109. let log_str = c_str.to_str().unwrap();
  110. // Don't change the mapping relation between number and level
  111. match level {
  112. 0 => tracing::info!("{}", log_str),
  113. 1 => tracing::debug!("{}", log_str),
  114. 2 => tracing::trace!("{}", log_str),
  115. 3 => tracing::warn!("{}", log_str),
  116. 4 => tracing::error!("{}", log_str),
  117. _ => (),
  118. }
  119. }