Merge pull request #4186 from fufesou/feat/plugin_framework
plugin_framework
This commit is contained in:
commit
1cc02ef7e0
@ -1466,7 +1466,11 @@ class PluginCardState extends State<PluginCard> {
|
|||||||
final children = [
|
final children = [
|
||||||
_Button(
|
_Button(
|
||||||
'Reload',
|
'Reload',
|
||||||
() => bind.pluginReload(id: widget.pluginId),
|
() async {
|
||||||
|
clearPlugin(widget.pluginId);
|
||||||
|
await bind.pluginReload(id: widget.pluginId);
|
||||||
|
setState(() {});
|
||||||
|
},
|
||||||
),
|
),
|
||||||
_Checkbox(
|
_Checkbox(
|
||||||
label: 'Enable',
|
label: 'Enable',
|
||||||
@ -1476,6 +1480,7 @@ class PluginCardState extends State<PluginCard> {
|
|||||||
clearPlugin(widget.pluginId);
|
clearPlugin(widget.pluginId);
|
||||||
}
|
}
|
||||||
await bind.pluginIdEnable(id: widget.pluginId, v: v);
|
await bind.pluginIdEnable(id: widget.pluginId, v: v);
|
||||||
|
setState(() {});
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
];
|
];
|
||||||
|
@ -51,6 +51,11 @@ class LocationModel with ChangeNotifier {
|
|||||||
notifyListeners();
|
notifyListeners();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void remove(PluginId id) {
|
||||||
|
pluginModels.remove(id);
|
||||||
|
notifyListeners();
|
||||||
|
}
|
||||||
|
|
||||||
bool get isEmpty => pluginModels.isEmpty;
|
bool get isEmpty => pluginModels.isEmpty;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -68,7 +73,7 @@ PluginModel? getPluginModel(String location, PluginId id) =>
|
|||||||
|
|
||||||
void clearPlugin(PluginId pluginId) {
|
void clearPlugin(PluginId pluginId) {
|
||||||
for (var element in _locationModels.values) {
|
for (var element in _locationModels.values) {
|
||||||
element.pluginModels.remove(pluginId);
|
element.remove(pluginId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,7 +60,8 @@ struct MsgToConfig {
|
|||||||
/// id: The id of this plugin.
|
/// id: The id of this plugin.
|
||||||
/// content: The content.
|
/// content: The content.
|
||||||
/// len: The length of the content.
|
/// len: The length of the content.
|
||||||
pub fn cb_msg(
|
#[no_mangle]
|
||||||
|
pub extern "C" fn cb_msg(
|
||||||
peer: *const c_char,
|
peer: *const c_char,
|
||||||
target: *const c_char,
|
target: *const c_char,
|
||||||
id: *const c_char,
|
id: *const c_char,
|
||||||
|
@ -20,6 +20,7 @@ lazy_static::lazy_static! {
|
|||||||
Arc::new(Mutex::new(conf))
|
Arc::new(Mutex::new(conf))
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
use crate::ui_interface::get_id;
|
||||||
|
|
||||||
pub(super) const CONFIG_TYPE_SHARED: &str = "shared";
|
pub(super) const CONFIG_TYPE_SHARED: &str = "shared";
|
||||||
pub(super) const CONFIG_TYPE_PEER: &str = "peer";
|
pub(super) const CONFIG_TYPE_PEER: &str = "peer";
|
||||||
@ -343,8 +344,12 @@ impl ManagerConfig {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(super) extern "C" fn cb_get_local_peer_id() -> *const c_char {
|
||||||
|
str_to_cstr_ret(&get_id())
|
||||||
|
}
|
||||||
|
|
||||||
// Return shared config if peer is nullptr.
|
// Return shared config if peer is nullptr.
|
||||||
pub(super) fn cb_get_conf(
|
pub(super) extern "C" fn cb_get_conf(
|
||||||
peer: *const c_char,
|
peer: *const c_char,
|
||||||
id: *const c_char,
|
id: *const c_char,
|
||||||
key: *const c_char,
|
key: *const c_char,
|
||||||
|
@ -5,6 +5,7 @@ mod callback_msg;
|
|||||||
mod config;
|
mod config;
|
||||||
pub mod desc;
|
pub mod desc;
|
||||||
mod errno;
|
mod errno;
|
||||||
|
mod plog;
|
||||||
pub mod ipc;
|
pub mod ipc;
|
||||||
mod plugins;
|
mod plugins;
|
||||||
|
|
||||||
|
34
src/plugin/plog.rs
Normal file
34
src/plugin/plog.rs
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
use hbb_common::log;
|
||||||
|
use std::ffi::c_char;
|
||||||
|
|
||||||
|
const LOG_LEVEL_TRACE: &[u8; 6] = b"trace\0";
|
||||||
|
const LOG_LEVEL_DEBUG: &[u8; 6] = b"debug\0";
|
||||||
|
const LOG_LEVEL_INFO: &[u8; 5] = b"info\0";
|
||||||
|
const LOG_LEVEL_WARN: &[u8; 5] = b"warn\0";
|
||||||
|
const LOG_LEVEL_ERROR: &[u8; 6] = b"error\0";
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn is_level(level: *const c_char, level_bytes: &[u8]) -> bool {
|
||||||
|
level_bytes == unsafe { std::slice::from_raw_parts(level as *const u8, level_bytes.len()) }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[no_mangle]
|
||||||
|
pub(super) extern "C" fn log(level: *const c_char, msg: *const c_char) {
|
||||||
|
if level.is_null() || msg.is_null() {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Ok(msg) = super::cstr_to_string(msg) {
|
||||||
|
if is_level(level, LOG_LEVEL_TRACE) {
|
||||||
|
log::trace!("{}", msg);
|
||||||
|
} else if is_level(level, LOG_LEVEL_DEBUG) {
|
||||||
|
log::debug!("{}", msg);
|
||||||
|
} else if is_level(level, LOG_LEVEL_INFO) {
|
||||||
|
log::info!("{}", msg);
|
||||||
|
} else if is_level(level, LOG_LEVEL_WARN) {
|
||||||
|
log::warn!("{}", msg);
|
||||||
|
} else if is_level(level, LOG_LEVEL_ERROR) {
|
||||||
|
log::error!("{}", msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,7 +1,7 @@
|
|||||||
use super::{desc::Desc, errno::*, *};
|
use super::{desc::Desc, errno::*, *};
|
||||||
#[cfg(not(debug_assertions))]
|
#[cfg(not(debug_assertions))]
|
||||||
use crate::common::is_server;
|
use crate::common::is_server;
|
||||||
use crate::{flutter, ui_interface::get_id};
|
use crate::flutter;
|
||||||
use hbb_common::{
|
use hbb_common::{
|
||||||
allow_err, bail,
|
allow_err, bail,
|
||||||
dlopen::symbor::Library,
|
dlopen::symbor::Library,
|
||||||
@ -21,8 +21,7 @@ const METHOD_HANDLE_PEER: &[u8; 12] = b"handle_peer\0";
|
|||||||
|
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
||||||
static ref PLUGIN_INFO: Arc<RwLock<HashMap<String, PluginInfo>>> = Default::default();
|
static ref PLUGIN_INFO: Arc<RwLock<HashMap<String, PluginInfo>>> = Default::default();
|
||||||
pub static ref PLUGINS: Arc<RwLock<HashMap<String, Plugin>>> = Default::default();
|
static ref PLUGINS: Arc<RwLock<HashMap<String, Plugin>>> = Default::default();
|
||||||
pub static ref LOCAL_PEER_ID: Arc<RwLock<String>> = Default::default();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct PluginInfo {
|
struct PluginInfo {
|
||||||
@ -32,25 +31,29 @@ struct PluginInfo {
|
|||||||
|
|
||||||
/// Initialize the plugins.
|
/// Initialize the plugins.
|
||||||
///
|
///
|
||||||
/// Return null ptr if success.
|
/// data: The initialize data.
|
||||||
/// Return the error message if failed. `i32-String` without dash, i32 is a signed little-endian number, the String is utf8 string.
|
|
||||||
/// The plugin allocate memory with `libc::malloc` and return the pointer.
|
|
||||||
pub type PluginFuncInit = fn() -> *const c_void;
|
|
||||||
/// Reset the plugin.
|
|
||||||
///
|
///
|
||||||
/// Return null ptr if success.
|
/// Return null ptr if success.
|
||||||
/// Return the error message if failed. `i32-String` without dash, i32 is a signed little-endian number, the String is utf8 string.
|
/// Return the error message if failed. `i32-String` without dash, i32 is a signed little-endian number, the String is utf8 string.
|
||||||
/// The plugin allocate memory with `libc::malloc` and return the pointer.
|
/// The plugin allocate memory with `libc::malloc` and return the pointer.
|
||||||
pub type PluginFuncReset = fn() -> *const c_void;
|
type PluginFuncInit = extern "C" fn(data: *const InitData) -> *const c_void;
|
||||||
|
/// Reset the plugin.
|
||||||
|
///
|
||||||
|
/// data: The initialize data.
|
||||||
|
///
|
||||||
|
/// Return null ptr if success.
|
||||||
|
/// Return the error message if failed. `i32-String` without dash, i32 is a signed little-endian number, the String is utf8 string.
|
||||||
|
/// The plugin allocate memory with `libc::malloc` and return the pointer.
|
||||||
|
type PluginFuncReset = extern "C" fn(data: *const InitData) -> *const c_void;
|
||||||
/// Clear the plugin.
|
/// Clear the plugin.
|
||||||
///
|
///
|
||||||
/// Return null ptr if success.
|
/// Return null ptr if success.
|
||||||
/// Return the error message if failed. `i32-String` without dash, i32 is a signed little-endian number, the String is utf8 string.
|
/// Return the error message if failed. `i32-String` without dash, i32 is a signed little-endian number, the String is utf8 string.
|
||||||
/// The plugin allocate memory with `libc::malloc` and return the pointer.
|
/// The plugin allocate memory with `libc::malloc` and return the pointer.
|
||||||
pub type PluginFuncClear = fn() -> *const c_void;
|
type PluginFuncClear = extern "C" fn() -> *const c_void;
|
||||||
/// Get the description of the plugin.
|
/// Get the description of the plugin.
|
||||||
/// Return the description. The plugin allocate memory with `libc::malloc` and return the pointer.
|
/// Return the description. The plugin allocate memory with `libc::malloc` and return the pointer.
|
||||||
pub type PluginFuncDesc = fn() -> *const c_char;
|
type PluginFuncDesc = extern "C" fn() -> *const c_char;
|
||||||
/// Callback to send message to peer or ui.
|
/// Callback to send message to peer or ui.
|
||||||
/// peer, target, id are utf8 strings(null terminated).
|
/// peer, target, id are utf8 strings(null terminated).
|
||||||
///
|
///
|
||||||
@ -59,17 +62,13 @@ pub type PluginFuncDesc = fn() -> *const c_char;
|
|||||||
/// id: The id of this plugin.
|
/// id: The id of this plugin.
|
||||||
/// content: The content.
|
/// content: The content.
|
||||||
/// len: The length of the content.
|
/// len: The length of the content.
|
||||||
type CallbackMsg = fn(
|
type CallbackMsg = extern "C" fn(
|
||||||
peer: *const c_char,
|
peer: *const c_char,
|
||||||
target: *const c_char,
|
target: *const c_char,
|
||||||
id: *const c_char,
|
id: *const c_char,
|
||||||
content: *const c_void,
|
content: *const c_void,
|
||||||
len: usize,
|
len: usize,
|
||||||
);
|
);
|
||||||
/// Callback to get the id of local peer id.
|
|
||||||
/// The returned string is utf8 string(null terminated).
|
|
||||||
/// Don't free the returned ptr.
|
|
||||||
type CallbackGetId = fn() -> *const c_char;
|
|
||||||
/// Callback to get the config.
|
/// Callback to get the config.
|
||||||
/// peer, key are utf8 strings(null terminated).
|
/// peer, key are utf8 strings(null terminated).
|
||||||
///
|
///
|
||||||
@ -79,7 +78,17 @@ type CallbackGetId = fn() -> *const c_char;
|
|||||||
///
|
///
|
||||||
/// The returned string is utf8 string(null terminated) and must be freed by caller.
|
/// The returned string is utf8 string(null terminated) and must be freed by caller.
|
||||||
type CallbackGetConf =
|
type CallbackGetConf =
|
||||||
fn(peer: *const c_char, id: *const c_char, key: *const c_char) -> *const c_char;
|
extern "C" fn(peer: *const c_char, id: *const c_char, key: *const c_char) -> *const c_char;
|
||||||
|
/// Get local peer id.
|
||||||
|
///
|
||||||
|
/// The returned string is utf8 string(null terminated) and must be freed by caller.
|
||||||
|
type CallbackGetId = extern "C" fn() -> *const c_char;
|
||||||
|
/// Callback to log.
|
||||||
|
///
|
||||||
|
/// level, msg are utf8 strings(null terminated).
|
||||||
|
/// level: "error", "warn", "info", "debug", "trace".
|
||||||
|
/// msg: The message.
|
||||||
|
type CallbackLog = extern "C" fn(level: *const c_char, msg: *const c_char);
|
||||||
/// The main function of the plugin.
|
/// The main function of the plugin.
|
||||||
/// method: The method. "handle_ui" or "handle_peer"
|
/// method: The method. "handle_ui" or "handle_peer"
|
||||||
/// peer: The peer id.
|
/// peer: The peer id.
|
||||||
@ -88,20 +97,41 @@ type CallbackGetConf =
|
|||||||
/// Return null ptr if success.
|
/// Return null ptr if success.
|
||||||
/// Return the error message if failed. `i32-String` without dash, i32 is a signed little-endian number, the String is utf8 string.
|
/// Return the error message if failed. `i32-String` without dash, i32 is a signed little-endian number, the String is utf8 string.
|
||||||
/// The plugin allocate memory with `libc::malloc` and return the pointer.
|
/// The plugin allocate memory with `libc::malloc` and return the pointer.
|
||||||
pub type PluginFuncCall = fn(
|
type PluginFuncCall = extern "C" fn(
|
||||||
method: *const c_char,
|
method: *const c_char,
|
||||||
peer: *const c_char,
|
peer: *const c_char,
|
||||||
args: *const c_void,
|
args: *const c_void,
|
||||||
len: usize,
|
len: usize,
|
||||||
) -> *const c_void;
|
) -> *const c_void;
|
||||||
|
|
||||||
|
/// The plugin callbacks.
|
||||||
|
/// msg: The callback to send message to peer or ui.
|
||||||
|
/// get_conf: The callback to get the config.
|
||||||
|
/// log: The callback to log.
|
||||||
#[repr(C)]
|
#[repr(C)]
|
||||||
|
#[derive(Copy, Clone)]
|
||||||
struct Callbacks {
|
struct Callbacks {
|
||||||
msg: CallbackMsg,
|
msg: CallbackMsg,
|
||||||
get_id: CallbackGetId,
|
|
||||||
get_conf: CallbackGetConf,
|
get_conf: CallbackGetConf,
|
||||||
|
get_id: CallbackGetId,
|
||||||
|
log: CallbackLog,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// The plugin initialize data.
|
||||||
|
/// version: The version of the plugin, can't be nullptr.
|
||||||
|
/// local_peer_id: The local peer id, can't be nullptr.
|
||||||
|
/// cbs: The callbacks.
|
||||||
|
#[repr(C)]
|
||||||
|
struct InitData {
|
||||||
|
version: *const c_char,
|
||||||
|
cbs: Callbacks,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for InitData {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
free_c_ptr(self.version as _);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
type PluginFuncSetCallbacks = fn(Callbacks);
|
|
||||||
|
|
||||||
macro_rules! make_plugin {
|
macro_rules! make_plugin {
|
||||||
($($field:ident : $tp:ty),+) => {
|
($($field:ident : $tp:ty),+) => {
|
||||||
@ -148,8 +178,8 @@ macro_rules! make_plugin {
|
|||||||
desc
|
desc
|
||||||
}
|
}
|
||||||
|
|
||||||
fn init(&self, path: &str) -> ResultType<()> {
|
fn init(&self, data: &InitData, path: &str) -> ResultType<()> {
|
||||||
let init_ret = (self.init)();
|
let init_ret = (self.init)(data as _);
|
||||||
if !init_ret.is_null() {
|
if !init_ret.is_null() {
|
||||||
let (code, msg) = get_code_msg_from_ret(init_ret);
|
let (code, msg) = get_code_msg_from_ret(init_ret);
|
||||||
free_c_ptr(init_ret as _);
|
free_c_ptr(init_ret as _);
|
||||||
@ -192,8 +222,7 @@ make_plugin!(
|
|||||||
reset: PluginFuncReset,
|
reset: PluginFuncReset,
|
||||||
clear: PluginFuncClear,
|
clear: PluginFuncClear,
|
||||||
desc: PluginFuncDesc,
|
desc: PluginFuncDesc,
|
||||||
call: PluginFuncCall,
|
call: PluginFuncCall
|
||||||
set_cbs: PluginFuncSetCallbacks
|
|
||||||
);
|
);
|
||||||
|
|
||||||
#[cfg(target_os = "windows")]
|
#[cfg(target_os = "windows")]
|
||||||
@ -260,21 +289,6 @@ pub fn reload_plugin(id: &str) -> ResultType<()> {
|
|||||||
load_plugin(Some(&path), Some(id))
|
load_plugin(Some(&path), Some(id))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[no_mangle]
|
|
||||||
fn cb_get_local_peer_id() -> *const c_char {
|
|
||||||
let mut id = (*LOCAL_PEER_ID.read().unwrap()).clone();
|
|
||||||
if id.is_empty() {
|
|
||||||
let mut lock = LOCAL_PEER_ID.write().unwrap();
|
|
||||||
id = (*lock).clone();
|
|
||||||
if id.is_empty() {
|
|
||||||
id = get_id();
|
|
||||||
id.push('\0');
|
|
||||||
*lock = id.clone();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
id.as_ptr() as _
|
|
||||||
}
|
|
||||||
|
|
||||||
fn load_plugin_path(path: &str) -> ResultType<()> {
|
fn load_plugin_path(path: &str) -> ResultType<()> {
|
||||||
let plugin = Plugin::new(path)?;
|
let plugin = Plugin::new(path)?;
|
||||||
let desc = plugin.desc()?;
|
let desc = plugin.desc()?;
|
||||||
@ -282,19 +296,21 @@ fn load_plugin_path(path: &str) -> ResultType<()> {
|
|||||||
// to-do validate plugin
|
// to-do validate plugin
|
||||||
// to-do check the plugin id (make sure it does not use another plugin's id)
|
// to-do check the plugin id (make sure it does not use another plugin's id)
|
||||||
|
|
||||||
plugin.init(path)?;
|
let init_data = InitData {
|
||||||
|
version: str_to_cstr_ret(crate::VERSION),
|
||||||
|
cbs: Callbacks {
|
||||||
|
msg: callback_msg::cb_msg,
|
||||||
|
get_conf: config::cb_get_conf,
|
||||||
|
get_id: config::cb_get_local_peer_id,
|
||||||
|
log: super::plog::log,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
plugin.init(&init_data, path)?;
|
||||||
|
|
||||||
if change_manager() {
|
if change_manager() {
|
||||||
super::config::ManagerConfig::add_plugin(desc.id())?;
|
super::config::ManagerConfig::add_plugin(desc.id())?;
|
||||||
}
|
}
|
||||||
|
|
||||||
// set callbacks
|
|
||||||
(plugin.set_cbs)(Callbacks {
|
|
||||||
msg: callback_msg::cb_msg,
|
|
||||||
get_id: cb_get_local_peer_id,
|
|
||||||
get_conf: config::cb_get_conf,
|
|
||||||
});
|
|
||||||
|
|
||||||
// update ui
|
// update ui
|
||||||
// Ui may be not ready now, so we need to update again once ui is ready.
|
// Ui may be not ready now, so we need to update again once ui is ready.
|
||||||
update_ui_plugin_desc(&desc, None);
|
update_ui_plugin_desc(&desc, None);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user