lib.rs 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  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 FLOWY_SDK: RwLock<Option<FlowySDK>> = 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 = FlowySDKConfig::new(path, "appflowy".to_string(), server_config).log_filter("info", log_crates);
  30. *FLOWY_SDK.write() = Some(FlowySDK::new(config));
  31. 0
  32. }
  33. #[no_mangle]
  34. pub extern "C" fn async_event(port: i64, input: *const u8, len: usize) {
  35. let request: AFPluginRequest = FFIRequest::from_u8_pointer(input, len).into();
  36. log::trace!(
  37. "[FFI]: {} Async Event: {:?} with {} port",
  38. &request.id,
  39. &request.event,
  40. port
  41. );
  42. let dispatcher = match FLOWY_SDK.read().as_ref() {
  43. None => {
  44. log::error!("sdk not init yet.");
  45. return;
  46. }
  47. Some(e) => e.event_dispatcher.clone(),
  48. };
  49. let _ = AFPluginDispatcher::async_send_with_callback(dispatcher, request, move |resp: AFPluginEventResponse| {
  50. log::trace!("[FFI]: Post data to dart through {} port", port);
  51. Box::pin(post_to_flutter(resp, port))
  52. });
  53. }
  54. #[no_mangle]
  55. pub extern "C" fn sync_event(input: *const u8, len: usize) -> *const u8 {
  56. let request: AFPluginRequest = FFIRequest::from_u8_pointer(input, len).into();
  57. log::trace!("[FFI]: {} Sync Event: {:?}", &request.id, &request.event,);
  58. let dispatcher = match FLOWY_SDK.read().as_ref() {
  59. None => {
  60. log::error!("sdk not init yet.");
  61. return forget_rust(Vec::default());
  62. }
  63. Some(e) => e.event_dispatcher.clone(),
  64. };
  65. let _response = AFPluginDispatcher::sync_send(dispatcher, request);
  66. // FFIResponse { }
  67. let response_bytes = vec![];
  68. let result = extend_front_four_bytes_into_bytes(&response_bytes);
  69. forget_rust(result)
  70. }
  71. #[no_mangle]
  72. pub extern "C" fn set_stream_port(port: i64) -> i32 {
  73. register_notification_sender(DartNotificationSender::new(port));
  74. 0
  75. }
  76. #[inline(never)]
  77. #[no_mangle]
  78. pub extern "C" fn link_me_please() {}
  79. #[inline(always)]
  80. async fn post_to_flutter(response: AFPluginEventResponse, port: i64) {
  81. let isolate = allo_isolate::Isolate::new(port);
  82. match isolate
  83. .catch_unwind(async {
  84. let ffi_resp = FFIResponse::from(response);
  85. ffi_resp.into_bytes().unwrap().to_vec()
  86. })
  87. .await
  88. {
  89. Ok(_success) => {
  90. log::trace!("[FFI]: Post data to dart success");
  91. }
  92. Err(e) => {
  93. if let Some(msg) = e.downcast_ref::<&str>() {
  94. log::error!("[FFI]: {:?}", msg);
  95. } else {
  96. log::error!("[FFI]: allo_isolate post panic");
  97. }
  98. }
  99. }
  100. }