mirror of
https://github.com/AppFlowy-IO/AppFlowy.git
synced 2024-08-30 18:12:39 +00:00
add dart ffi
This commit is contained in:
parent
97dde2d711
commit
42b8add240
@ -26,10 +26,12 @@ thread-id = "3.3.0"
|
|||||||
allo-isolate = {version = "^0.1", features = ["catch-unwind",], optional = true}
|
allo-isolate = {version = "^0.1", features = ["catch-unwind",], optional = true}
|
||||||
byteorder = {version = "1.3.4", optional = true}
|
byteorder = {version = "1.3.4", optional = true}
|
||||||
ffi-support = {version = "0.4.2", optional = true}
|
ffi-support = {version = "0.4.2", optional = true}
|
||||||
|
protobuf = {version = "2.20.0", optional = true}
|
||||||
|
lazy_static = {version = "1.4.0", optional = true}
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
tokio = { version = "1", features = ["full"] }
|
tokio = { version = "1", features = ["full"] }
|
||||||
futures-util = "0.3.15"
|
futures-util = "0.3.15"
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
dart_ffi = ["ffi-support", "allo-isolate", "byteorder"]
|
dart_ffi = ["ffi-support", "allo-isolate", "byteorder", "protobuf", "lazy_static"]
|
@ -1,13 +1,33 @@
|
|||||||
use crate::{response::EventResponse, rt::SystemCommand};
|
use crate::{
|
||||||
|
module::Module,
|
||||||
|
request::EventRequest,
|
||||||
|
response::EventResponse,
|
||||||
|
rt::SystemCommand,
|
||||||
|
stream::*,
|
||||||
|
system::FlowySystem,
|
||||||
|
};
|
||||||
use futures_core::ready;
|
use futures_core::ready;
|
||||||
use std::{future::Future, task::Context};
|
use lazy_static::lazy_static;
|
||||||
|
use protobuf::Message;
|
||||||
|
use std::{
|
||||||
|
cell::RefCell,
|
||||||
|
future::Future,
|
||||||
|
sync::{Arc, RwLock},
|
||||||
|
task::Context,
|
||||||
|
};
|
||||||
use tokio::{
|
use tokio::{
|
||||||
macros::support::{Pin, Poll},
|
macros::support::{Pin, Poll},
|
||||||
sync::{mpsc::UnboundedReceiver, oneshot},
|
sync::{mpsc::UnboundedSender, oneshot},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub extern "C" fn async_command(port: i64, input: *const u8, len: usize) {}
|
pub extern "C" fn async_command(port: i64, input: *const u8, len: usize) {
|
||||||
|
let bytes = unsafe { std::slice::from_raw_parts(input, len) }.to_vec();
|
||||||
|
let request = EventRequest::from_data(bytes);
|
||||||
|
|
||||||
|
let stream_data = StreamData::new(port, Some(request), Box::new(|port, response| {}));
|
||||||
|
send(stream_data);
|
||||||
|
}
|
||||||
|
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub extern "C" fn free_rust(ptr: *mut u8, length: u32) { reclaim_rust(ptr, length) }
|
pub extern "C" fn free_rust(ptr: *mut u8, length: u32) { reclaim_rust(ptr, length) }
|
||||||
@ -15,20 +35,59 @@ pub extern "C" fn free_rust(ptr: *mut u8, length: u32) { reclaim_rust(ptr, lengt
|
|||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub extern "C" fn init_stream(port: i64) -> i32 { return 0; }
|
pub extern "C" fn init_stream(port: i64) -> i32 { return 0; }
|
||||||
|
|
||||||
struct SystemFFI {
|
#[allow(unused_attributes)]
|
||||||
resp_rx: UnboundedReceiver<EventResponse>,
|
pub fn reclaim_rust(ptr: *mut u8, length: u32) {
|
||||||
|
unsafe {
|
||||||
|
let len: usize = length as usize;
|
||||||
|
Vec::from_raw_parts(ptr, len, len);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Future for SystemFFI {
|
thread_local!(
|
||||||
type Output = ();
|
static STREAM_SENDER: RefCell<Option<UnboundedSender<StreamData<i64>>>> = RefCell::new(None);
|
||||||
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
|
);
|
||||||
loop {
|
|
||||||
match ready!(Pin::new(&mut self.resp_rx).poll_recv(cx)) {
|
pub fn send(data: StreamData<i64>) {
|
||||||
None => return Poll::Ready(()),
|
STREAM_SENDER.with(|cell| match &*cell.borrow() {
|
||||||
Some(resp) => {
|
Some(tx) => {
|
||||||
log::trace!("Response: {:?}", resp);
|
tx.send(data);
|
||||||
},
|
},
|
||||||
|
None => panic!(""),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
pub fn init_dart<F>(modules: Vec<Module>, f: F)
|
||||||
|
where
|
||||||
|
F: FnOnce() + 'static,
|
||||||
|
{
|
||||||
|
let mut stream = CommandStream::<i64>::new();
|
||||||
|
let stream = CommandStream::<i64>::new();
|
||||||
|
let tx = stream.tx();
|
||||||
|
|
||||||
|
STREAM_SENDER.with(|cell| {
|
||||||
|
*cell.borrow_mut() = Some(tx);
|
||||||
|
});
|
||||||
|
|
||||||
|
FlowySystem::construct(|| modules, stream)
|
||||||
|
.spawn(async { f() })
|
||||||
|
.run()
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
// FlowySystem::construct(|| modules, stream)
|
||||||
|
// .spawn(async move {
|
||||||
|
// let request = EventRequest::new("1".to_string());
|
||||||
|
// let stream_data = StreamData::new(
|
||||||
|
// 1,
|
||||||
|
// Some(request),
|
||||||
|
// Box::new(|config, response| {
|
||||||
|
// log::info!("😁{:?}", response);
|
||||||
|
// }),
|
||||||
|
// );
|
||||||
|
//
|
||||||
|
// send(stream_data);
|
||||||
|
//
|
||||||
|
// FlowySystem::current().stop();
|
||||||
|
// })
|
||||||
|
// .run()
|
||||||
|
// .unwrap();
|
||||||
}
|
}
|
||||||
|
@ -1,2 +1,4 @@
|
|||||||
#[cfg(feature = "dart_ffi")]
|
#[cfg(feature = "dart_ffi")]
|
||||||
mod ffi;
|
mod ffi;
|
||||||
|
|
||||||
|
pub use ffi::*;
|
||||||
|
@ -10,7 +10,7 @@ mod service;
|
|||||||
mod util;
|
mod util;
|
||||||
|
|
||||||
#[cfg(feature = "dart_ffi")]
|
#[cfg(feature = "dart_ffi")]
|
||||||
mod dart_ffi;
|
pub mod dart_ffi;
|
||||||
mod stream;
|
mod stream;
|
||||||
mod system;
|
mod system;
|
||||||
|
|
||||||
|
@ -21,6 +21,7 @@ use std::{
|
|||||||
future::Future,
|
future::Future,
|
||||||
pin::Pin,
|
pin::Pin,
|
||||||
rc::Rc,
|
rc::Rc,
|
||||||
|
sync::Arc,
|
||||||
task::{Context, Poll},
|
task::{Context, Poll},
|
||||||
};
|
};
|
||||||
use tokio::sync::mpsc::{unbounded_channel, UnboundedReceiver, UnboundedSender};
|
use tokio::sync::mpsc::{unbounded_channel, UnboundedReceiver, UnboundedSender};
|
||||||
|
@ -30,6 +30,8 @@ impl EventRequest {
|
|||||||
pub fn get_event(&self) -> &str { &self.event }
|
pub fn get_event(&self) -> &str { &self.event }
|
||||||
|
|
||||||
pub fn get_id(&self) -> &str { &self.id }
|
pub fn get_id(&self) -> &str { &self.id }
|
||||||
|
|
||||||
|
pub fn from_data(data: Vec<u8>) -> Self { unimplemented!() }
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait FromRequest: Sized {
|
pub trait FromRequest: Sized {
|
||||||
|
@ -4,7 +4,7 @@ use crate::{
|
|||||||
request::EventRequest,
|
request::EventRequest,
|
||||||
response::EventResponse,
|
response::EventResponse,
|
||||||
service::{BoxService, Service, ServiceFactory},
|
service::{BoxService, Service, ServiceFactory},
|
||||||
system::ModuleMap,
|
system::ModuleServiceMap,
|
||||||
};
|
};
|
||||||
use futures_core::{future::LocalBoxFuture, ready, task::Context};
|
use futures_core::{future::LocalBoxFuture, ready, task::Context};
|
||||||
use std::{collections::HashMap, future::Future, rc::Rc};
|
use std::{collections::HashMap, future::Future, rc::Rc};
|
||||||
@ -13,7 +13,7 @@ use tokio::{
|
|||||||
sync::mpsc::{unbounded_channel, UnboundedReceiver, UnboundedSender},
|
sync::mpsc::{unbounded_channel, UnboundedReceiver, UnboundedSender},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub type BoxStreamCallback<T> = Box<dyn FnOnce(T, EventResponse) + 'static>;
|
pub type BoxStreamCallback<T> = Box<dyn FnOnce(T, EventResponse) + 'static + Send + Sync>;
|
||||||
pub struct StreamData<T>
|
pub struct StreamData<T>
|
||||||
where
|
where
|
||||||
T: 'static,
|
T: 'static,
|
||||||
@ -37,22 +37,26 @@ pub struct CommandStream<T>
|
|||||||
where
|
where
|
||||||
T: 'static,
|
T: 'static,
|
||||||
{
|
{
|
||||||
module_map: ModuleMap,
|
module_map: Option<ModuleServiceMap>,
|
||||||
pub data_tx: UnboundedSender<StreamData<T>>,
|
data_tx: UnboundedSender<StreamData<T>>,
|
||||||
data_rx: UnboundedReceiver<StreamData<T>>,
|
data_rx: UnboundedReceiver<StreamData<T>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> CommandStream<T> {
|
impl<T> CommandStream<T> {
|
||||||
pub fn new(module_map: ModuleMap) -> Self {
|
pub fn new() -> Self {
|
||||||
let (data_tx, data_rx) = unbounded_channel::<StreamData<T>>();
|
let (data_tx, data_rx) = unbounded_channel::<StreamData<T>>();
|
||||||
Self {
|
Self {
|
||||||
module_map,
|
module_map: None,
|
||||||
data_tx,
|
data_tx,
|
||||||
data_rx,
|
data_rx,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn send(&self, data: StreamData<T>) { let _ = self.data_tx.send(data); }
|
pub fn send(&self, data: StreamData<T>) { let _ = self.data_tx.send(data); }
|
||||||
|
|
||||||
|
pub fn module_service_map(&mut self, map: ModuleServiceMap) { self.module_map = Some(map) }
|
||||||
|
|
||||||
|
pub fn tx(&self) -> UnboundedSender<StreamData<T>> { self.data_tx.clone() }
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> Future for CommandStream<T>
|
impl<T> Future for CommandStream<T>
|
||||||
@ -87,14 +91,14 @@ where
|
|||||||
type Future = LocalBoxFuture<'static, Result<Self::Service, Self::Error>>;
|
type Future = LocalBoxFuture<'static, Result<Self::Service, Self::Error>>;
|
||||||
|
|
||||||
fn new_service(&self, _cfg: Self::Config) -> Self::Future {
|
fn new_service(&self, _cfg: Self::Config) -> Self::Future {
|
||||||
let module_map = self.module_map.clone();
|
let module_map = self.module_map.as_ref().unwrap().clone();
|
||||||
let service = Box::new(CommandStreamService { module_map });
|
let service = Box::new(CommandStreamService { module_map });
|
||||||
Box::pin(async move { Ok(service as Self::Service) })
|
Box::pin(async move { Ok(service as Self::Service) })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct CommandStreamService {
|
pub struct CommandStreamService {
|
||||||
module_map: ModuleMap,
|
module_map: ModuleServiceMap,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T> Service<StreamData<T>> for CommandStreamService
|
impl<T> Service<StreamData<T>> for CommandStreamService
|
||||||
@ -118,12 +122,18 @@ where
|
|||||||
let service_fut = fut.await?.call(request);
|
let service_fut = fut.await?.call(request);
|
||||||
service_fut.await
|
service_fut.await
|
||||||
},
|
},
|
||||||
None => Err(InternalError::new("".to_owned()).into()),
|
None => {
|
||||||
|
let msg = format!("Can not find the module to handle the request:{:?}", request);
|
||||||
|
Err(InternalError::new(msg).into())
|
||||||
|
},
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let resp = result().await.unwrap();
|
match result().await {
|
||||||
(data.callback)(data.config, resp);
|
Ok(resp) => (data.callback)(data.config, resp),
|
||||||
|
Err(e) => log::error!("{:?}", e),
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
};
|
};
|
||||||
Box::pin(fut)
|
Box::pin(fut)
|
||||||
|
@ -4,7 +4,7 @@ use crate::{
|
|||||||
request::EventRequest,
|
request::EventRequest,
|
||||||
response::EventResponse,
|
response::EventResponse,
|
||||||
rt::Runtime,
|
rt::Runtime,
|
||||||
stream::CommandStreamService,
|
stream::{CommandStream, CommandStreamService, StreamData},
|
||||||
};
|
};
|
||||||
use futures_core::{ready, task::Context};
|
use futures_core::{ready, task::Context};
|
||||||
use std::{cell::RefCell, collections::HashMap, future::Future, io, rc::Rc, sync::Arc};
|
use std::{cell::RefCell, collections::HashMap, future::Future, io, rc::Rc, sync::Arc};
|
||||||
@ -26,14 +26,14 @@ pub enum SystemCommand {
|
|||||||
Response(EventResponse),
|
Response(EventResponse),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type ModuleMap = Rc<HashMap<Event, Rc<Module>>>;
|
pub type ModuleServiceMap = Rc<HashMap<Event, Rc<Module>>>;
|
||||||
pub struct FlowySystem {
|
pub struct FlowySystem {
|
||||||
sys_cmd_tx: UnboundedSender<SystemCommand>,
|
sys_cmd_tx: UnboundedSender<SystemCommand>,
|
||||||
module_map: ModuleMap,
|
module_map: ModuleServiceMap,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FlowySystem {
|
impl FlowySystem {
|
||||||
pub fn construct<F>(module_factory: F) -> SystemRunner
|
pub fn construct<F, T>(module_factory: F, mut stream: CommandStream<T>) -> SystemRunner
|
||||||
where
|
where
|
||||||
F: FnOnce() -> Vec<Module>,
|
F: FnOnce() -> Vec<Module>,
|
||||||
{
|
{
|
||||||
@ -46,21 +46,26 @@ impl FlowySystem {
|
|||||||
sys_cmd_rx,
|
sys_cmd_rx,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let factory = module_factory();
|
||||||
|
let mut module_service_map = HashMap::new();
|
||||||
|
factory.into_iter().for_each(|m| {
|
||||||
|
let events = m.events();
|
||||||
|
let rc_module = Rc::new(m);
|
||||||
|
events.into_iter().for_each(|e| {
|
||||||
|
module_service_map.insert(e, rc_module.clone());
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
let mut system = Self {
|
let mut system = Self {
|
||||||
sys_cmd_tx: sys_cmd_tx.clone(),
|
sys_cmd_tx: sys_cmd_tx.clone(),
|
||||||
module_map: Rc::new(HashMap::default()),
|
module_map: Rc::new(HashMap::default()),
|
||||||
};
|
};
|
||||||
|
|
||||||
let factory = module_factory();
|
let map = Rc::new(module_service_map);
|
||||||
let mut module_map = HashMap::new();
|
system.module_map = map.clone();
|
||||||
factory.into_iter().for_each(|m| {
|
stream.module_service_map(map.clone());
|
||||||
let events = m.events();
|
|
||||||
let rc_module = Rc::new(m);
|
runtime.spawn(stream);
|
||||||
events.into_iter().for_each(|e| {
|
|
||||||
module_map.insert(e, rc_module.clone());
|
|
||||||
});
|
|
||||||
});
|
|
||||||
system.module_map = Rc::new(module_map);
|
|
||||||
|
|
||||||
FlowySystem::set_current(system);
|
FlowySystem::set_current(system);
|
||||||
let runner = SystemRunner { rt: runtime, stop_rx };
|
let runner = SystemRunner { rt: runtime, stop_rx };
|
||||||
@ -76,7 +81,7 @@ impl FlowySystem {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn module_map(&self) -> ModuleMap { self.module_map.clone() }
|
pub fn module_map(&self) -> ModuleServiceMap { self.module_map.clone() }
|
||||||
|
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
pub fn set_current(sys: FlowySystem) {
|
pub fn set_current(sys: FlowySystem) {
|
||||||
|
34
rust-lib/flowy-sys/tests/api/dart_ffi.rs
Normal file
34
rust-lib/flowy-sys/tests/api/dart_ffi.rs
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
use crate::helper::*;
|
||||||
|
use flowy_sys::{dart_ffi::*, prelude::*};
|
||||||
|
|
||||||
|
pub async fn no_params() -> String { "no params function call".to_string() }
|
||||||
|
pub async fn one_params(_s: String) -> String { "one params function call".to_string() }
|
||||||
|
pub async fn two_params(_s1: String, _s2: String) -> String { "two params function call".to_string() }
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_init() {
|
||||||
|
setup_env();
|
||||||
|
|
||||||
|
let no_params_command = "no params".to_string();
|
||||||
|
let one_params_command = "one params".to_string();
|
||||||
|
let two_params_command = "two params".to_string();
|
||||||
|
|
||||||
|
let modules = vec![Module::new()
|
||||||
|
.event(no_params_command.clone(), no_params)
|
||||||
|
.event(one_params_command.clone(), one_params)
|
||||||
|
.event(two_params_command.clone(), two_params)];
|
||||||
|
|
||||||
|
init_dart(modules, || {
|
||||||
|
let request = EventRequest::new(no_params_command);
|
||||||
|
let stream_data = StreamData::new(
|
||||||
|
1,
|
||||||
|
Some(request),
|
||||||
|
Box::new(|config, response| {
|
||||||
|
log::info!("😁😁😁 {:?}", response);
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
send(stream_data);
|
||||||
|
FlowySystem::current().stop();
|
||||||
|
});
|
||||||
|
}
|
@ -1,2 +1,3 @@
|
|||||||
|
mod dart_ffi;
|
||||||
mod helper;
|
mod helper;
|
||||||
mod module_event;
|
mod module_event;
|
||||||
|
@ -13,18 +13,17 @@ fn test() {
|
|||||||
let one_params_command = "one params".to_string();
|
let one_params_command = "one params".to_string();
|
||||||
let two_params_command = "two params".to_string();
|
let two_params_command = "two params".to_string();
|
||||||
|
|
||||||
let runner = FlowySystem::construct(|| {
|
let stream = CommandStream::<i64>::new();
|
||||||
|
let tx = stream.tx();
|
||||||
|
FlowySystem::construct(
|
||||||
|
|| {
|
||||||
vec![Module::new()
|
vec![Module::new()
|
||||||
.event(no_params_command.clone(), no_params)
|
.event(no_params_command.clone(), no_params)
|
||||||
.event(one_params_command.clone(), one_params)
|
.event(one_params_command.clone(), one_params)
|
||||||
.event(two_params_command.clone(), two_params)]
|
.event(two_params_command.clone(), two_params)]
|
||||||
});
|
},
|
||||||
|
stream,
|
||||||
let stream = CommandStream::new(FlowySystem::current().module_map());
|
)
|
||||||
let tx = stream.data_tx.clone();
|
|
||||||
|
|
||||||
runner
|
|
||||||
.spawn(stream)
|
|
||||||
.spawn(async move {
|
.spawn(async move {
|
||||||
let request = EventRequest::new(no_params_command.clone());
|
let request = EventRequest::new(no_params_command.clone());
|
||||||
let stream_data = StreamData::new(
|
let stream_data = StreamData::new(
|
||||||
|
Loading…
Reference in New Issue
Block a user