use crate::dart_isolate_wrapper::*; use crate::tools::*; use allo_isolate::*; use cfg_if::*; use ffi_support::*; use lazy_static::*; use opentelemetry::sdk::*; use opentelemetry::*; use opentelemetry_otlp::WithExportConfig; use serde::*; use std::os::raw::c_char; use std::sync::Arc; use tracing::*; use tracing_subscriber::prelude::*; use tracing_subscriber::*; // Globals lazy_static! { static ref VEILID_API: AsyncMutex> = AsyncMutex::new(None); } async fn get_veilid_api() -> Result { let api_lock = VEILID_API.lock().await; api_lock .as_ref() .cloned() .ok_or(veilid_core::VeilidAPIError::NotInitialized) } async fn take_veilid_api() -> Result { let mut api_lock = VEILID_API.lock().await; api_lock .take() .ok_or(veilid_core::VeilidAPIError::NotInitialized) } fn logfilter, V: AsRef<[T]>>(metadata: &Metadata, ignore_list: V) -> bool { // Skip filtered targets !match (metadata.target(), ignore_list.as_ref()) { (path, ignore) if !ignore.is_empty() => { // Check that the module path does not match any ignore filters ignore.iter().any(|v| path.starts_with(v.as_ref())) } _ => false, } } ///////////////////////////////////////// // FFI Helpers // Declare external routine to release ffi strings define_string_destructor!(free_string); // Utility types for async API results type APIResult = Result; const APIRESULT_VOID: APIResult<()> = APIResult::Ok(()); // Stream abort macro for simplified error handling macro_rules! check_err_json { ($stream:expr, $ex:expr) => { match $ex { Ok(v) => v, Err(e) => { $stream.abort_json(e); return; } } }; } ///////////////////////////////////////// // FFI-specific cofnig #[derive(Debug, Deserialize, Serialize)] pub struct VeilidFFIConfigLoggingTerminal { pub enabled: bool, pub level: veilid_core::VeilidLogLevel, } #[derive(Debug, Deserialize, Serialize)] pub struct VeilidFFIConfigLoggingOtlp { pub enabled: bool, pub level: veilid_core::VeilidLogLevel, pub grpc_endpoint: String, pub service_name: String, } #[derive(Debug, Deserialize, Serialize)] pub struct VeilidFFIConfigLogging { pub terminal: VeilidFFIConfigLoggingTerminal, pub otlp: VeilidFFIConfigLoggingOtlp, } #[derive(Debug, Deserialize, Serialize)] pub struct VeilidFFIConfig { pub logging: VeilidFFIConfigLogging, } ///////////////////////////////////////// // Initializer #[no_mangle] #[instrument] pub extern "C" fn initialize_veilid_flutter(dart_post_c_object_ptr: ffi::DartPostCObjectFnType) { unsafe { store_dart_post_cobject(dart_post_c_object_ptr); } use std::sync::Once; static INIT_BACKTRACE: Once = Once::new(); INIT_BACKTRACE.call_once(move || { std::env::set_var("RUST_BACKTRACE", "1"); std::panic::set_hook(Box::new(move |panic_info| { let (file, line) = if let Some(loc) = panic_info.location() { (loc.file(), loc.line()) } else { ("", 0) }; error!("### Rust `panic!` hit at file '{}', line {}", file, line); if let Some(s) = panic_info.payload().downcast_ref::<&str>() { error!("panic payload: {:?}", s); } else if let Some(s) = panic_info.payload().downcast_ref::() { error!("panic payload: {:?}", s); } else if let Some(a) = panic_info.payload().downcast_ref::() { error!("panic payload: {:?}", a); } else { error!("no panic payload"); } error!( " Complete stack trace:\n{:?}\n", backtrace::Backtrace::new() ); // And stop the process, no recovery is going to be possible here error!("aborting!"); std::process::abort(); })); }); } ////////////////////////////////////////////////////////////////////////////////// /// C-compatible FFI Functions #[no_mangle] #[instrument] pub extern "C" fn configure_veilid_platform(platform_config: FfiStr) { let platform_config = platform_config.into_opt_string(); let platform_config: VeilidFFIConfig = veilid_core::deserialize_opt_json(platform_config) .expect("failed to deserialize plaform config json"); // Set up subscriber and layers let mut ignore_list = Vec::::new(); for ig in veilid_core::DEFAULT_LOG_IGNORE_LIST { ignore_list.push(ig.to_owned()); } let subscriber = Registry::default(); // Terminal logger let subscriber = subscriber.with(if platform_config.logging.terminal.enabled { let terminal_max_log_level: level_filters::LevelFilter = platform_config .logging .terminal .level .to_tracing_level() .into(); let ignore_list = ignore_list.clone(); Some( fmt::Layer::new() .compact() .with_writer(std::io::stdout) .with_filter(terminal_max_log_level) .with_filter(filter::FilterFn::new(move |metadata| { logfilter(metadata, &ignore_list) })), ) } else { None }); // OpenTelemetry logger let subscriber = subscriber.with(if platform_config.logging.otlp.enabled { let otlp_max_log_level: level_filters::LevelFilter = platform_config.logging.otlp.level.to_tracing_level().into(); let grpc_endpoint = platform_config.logging.otlp.grpc_endpoint.clone(); cfg_if! { if #[cfg(feature="rt-async-std")] { let exporter = opentelemetry_otlp::new_exporter() .grpcio() .with_endpoint(grpc_endpoint); let batch = opentelemetry::runtime::AsyncStd; } else if #[cfg(feature="rt-tokio")] { let exporter = opentelemetry_otlp::new_exporter() .tonic() .with_endpoint(format!("http://{}", grpc_endpoint)); let batch = opentelemetry::runtime::Tokio; } } let tracer = opentelemetry_otlp::new_pipeline() .tracing() .with_exporter(exporter) .with_trace_config(opentelemetry::sdk::trace::config().with_resource( Resource::new(vec![KeyValue::new( opentelemetry_semantic_conventions::resource::SERVICE_NAME, format!( "{}:{}", platform_config.logging.otlp.service_name, hostname::get() .map(|s| s.to_string_lossy().into_owned()) .unwrap_or_else(|_| "unknown".to_owned())), )]), )) .install_batch(batch) .map_err(|e| format!("failed to install OpenTelemetry tracer: {}", e)) .unwrap(); let ignore_list = ignore_list.clone(); Some( tracing_opentelemetry::layer() .with_tracer(tracer) .with_filter(otlp_max_log_level) .with_filter(filter::FilterFn::new(move |metadata| { logfilter(metadata, &ignore_list) })), ) } else { None }); // API logger (always add layer, startup will init this if it is enabled in settings) let subscriber = subscriber.with(veilid_core::ApiTracingLayer::get()); subscriber .try_init() .map_err(|e| format!("failed to initialize logging: {}", e)) .expect("failed to initalize ffi platform"); } #[no_mangle] #[instrument] pub extern "C" fn startup_veilid_core(port: i64, config: FfiStr) { let config = config.into_opt_string(); let stream = DartIsolateStream::new(port); spawn(async move { let config_json = match config { Some(v) => v, None => { stream.abort_json(veilid_core::VeilidAPIError::MissingArgument { context: "startup_veilid_core".to_owned(), argument: "config".to_owned(), }); return; } }; let mut api_lock = VEILID_API.lock().await; if api_lock.is_some() { stream.abort_json(veilid_core::VeilidAPIError::AlreadyInitialized); return; } let sink = stream.clone(); let update_callback = Arc::new(move |update: veilid_core::VeilidUpdate| { let sink = sink.clone(); match update { veilid_core::VeilidUpdate::Shutdown => { sink.close(); } _ => { sink.item_json(update); } } }); let res = veilid_core::api_startup_json(update_callback, config_json).await; let veilid_api = check_err_json!(stream, res); *api_lock = Some(veilid_api); }); } #[no_mangle] pub extern "C" fn get_veilid_state(port: i64) { DartIsolateWrapper::new(port).spawn_result_json(async move { let veilid_api = get_veilid_api().await?; let core_state = veilid_api.get_state().await?; APIResult::Ok(core_state) }); } #[no_mangle] #[instrument(level = "debug")] pub extern "C" fn change_api_log_level(port: i64, log_level: FfiStr) { let log_level = log_level.into_opt_string(); DartIsolateWrapper::new(port).spawn_result_json(async move { let log_level: veilid_core::VeilidConfigLogLevel = veilid_core::deserialize_opt_json(log_level)?; //let veilid_api = get_veilid_api().await?; //veilid_api.change_api_log_level(log_level).await; veilid_core::ApiTracingLayer::change_api_log_level(log_level.to_veilid_log_level()); APIRESULT_VOID }); } #[no_mangle] #[instrument] pub extern "C" fn shutdown_veilid_core(port: i64) { DartIsolateWrapper::new(port).spawn_result_json(async move { let veilid_api = take_veilid_api().await?; veilid_api.shutdown().await; APIRESULT_VOID }); } #[no_mangle] pub extern "C" fn debug(port: i64, command: FfiStr) { let command = command.into_opt_string().unwrap_or_default(); DartIsolateWrapper::new(port).spawn_result(async move { let veilid_api = get_veilid_api().await?; let out = veilid_api.debug(command).await?; APIResult::Ok(out) }); } #[no_mangle] pub extern "C" fn veilid_version_string() -> *mut c_char { veilid_core::veilid_version_string().into_ffi_value() } #[repr(C)] pub struct VeilidVersion { pub major: u32, pub minor: u32, pub patch: u32, } #[no_mangle] #[instrument] pub extern "C" fn veilid_version() -> VeilidVersion { let (major, minor, patch) = veilid_core::veilid_version(); VeilidVersion { major, minor, patch, } }