lib.rs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. #![allow(clippy::not_unsafe_ptr_arg_deref)]
  2. use std::{ffi::CStr, os::raw::c_char};
  3. use lazy_static::lazy_static;
  4. use parking_lot::RwLock;
  5. use flowy_core::*;
  6. use flowy_notification::{register_notification_sender, unregister_all_notification_sender};
  7. use lib_dispatch::prelude::ToBytes;
  8. use lib_dispatch::prelude::*;
  9. use crate::env_serde::AppFlowyEnv;
  10. use crate::notification::DartNotificationSender;
  11. use crate::{
  12. c::{extend_front_four_bytes_into_bytes, forget_rust},
  13. model::{FFIRequest, FFIResponse},
  14. };
  15. mod c;
  16. mod env_serde;
  17. mod model;
  18. mod notification;
  19. mod protobuf;
  20. mod util;
  21. lazy_static! {
  22. static ref APPFLOWY_CORE: RwLock<Option<AppFlowyCore>> = RwLock::new(None);
  23. }
  24. #[no_mangle]
  25. pub extern "C" fn init_sdk(path: *mut c_char) -> i64 {
  26. let c_str: &CStr = unsafe { CStr::from_ptr(path) };
  27. let path: &str = c_str.to_str().unwrap();
  28. let log_crates = vec!["flowy-ffi".to_string()];
  29. let config =
  30. AppFlowyCoreConfig::new(path, DEFAULT_NAME.to_string()).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. 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. // Make sure hot reload won't register the notification sender twice
  79. unregister_all_notification_sender();
  80. register_notification_sender(DartNotificationSender::new(port));
  81. 0
  82. }
  83. #[inline(never)]
  84. #[no_mangle]
  85. pub extern "C" fn link_me_please() {}
  86. #[inline(always)]
  87. async fn post_to_flutter(response: AFPluginEventResponse, port: i64) {
  88. let isolate = allo_isolate::Isolate::new(port);
  89. match isolate
  90. .catch_unwind(async {
  91. let ffi_resp = FFIResponse::from(response);
  92. ffi_resp.into_bytes().unwrap().to_vec()
  93. })
  94. .await
  95. {
  96. Ok(_success) => {
  97. log::trace!("[FFI]: Post data to dart success");
  98. },
  99. Err(e) => {
  100. if let Some(msg) = e.downcast_ref::<&str>() {
  101. log::error!("[FFI]: {:?}", msg);
  102. } else {
  103. log::error!("[FFI]: allo_isolate post panic");
  104. }
  105. },
  106. }
  107. }
  108. #[no_mangle]
  109. pub extern "C" fn backend_log(level: i64, data: *const c_char) {
  110. let c_str = unsafe { CStr::from_ptr(data) };
  111. let log_str = c_str.to_str().unwrap();
  112. // Don't change the mapping relation between number and level
  113. match level {
  114. 0 => tracing::info!("{}", log_str),
  115. 1 => tracing::debug!("{}", log_str),
  116. 2 => tracing::trace!("{}", log_str),
  117. 3 => tracing::warn!("{}", log_str),
  118. 4 => tracing::error!("{}", log_str),
  119. _ => (),
  120. }
  121. }
  122. #[no_mangle]
  123. pub extern "C" fn set_env(data: *const c_char) {
  124. let c_str = unsafe { CStr::from_ptr(data) };
  125. let serde_str = c_str.to_str().unwrap();
  126. AppFlowyEnv::parser(serde_str);
  127. }