pvxs-sys 0.1.1

Low-level FFI bindings for EPICS PVXS library
Documentation
// Copyright 2026 Tine Zata
// SPDX-License-Identifier: MPL-2.0
//! # EPICS PVXS Rust Bindings
//!
//! Safe Rust bindings for the EPICS PVXS (PVAccess) library.
//!
//! ## Overview
//!
//! This crate provides idiomatic Rust bindings to the EPICS PVXS C++ library,
//! which implements the PVAccess network protocol used in EPICS (Experimental
//! Physics and Industrial Control System).
//!
//! ## Features
//!
//! ### Client
//! - **GET**: Read scalar and array PV values ([`Context::get`])
//! - **PUT**: Write double, int32, string, and enum scalars or arrays
//!   ([`Context::put_double`], [`Context::put_int32`], [`Context::put_string`],
//!   [`Context::put_enum`], and their `_array` variants)
//! - **Monitor**: Subscribe to real-time value changes via
//!   [`Context::monitor_builder`] → [`MonitorBuilder::connect_exception`] /
//!   [`MonitorBuilder::disconnect_exception`] / [`MonitorBuilder::event`] /
//!   [`MonitorBuilder::exec`] → [`Monitor::pop`]
//!
//! ### Server
//! - **Start**: [`Server::start_from_env`] (network) or [`Server::start_isolated`] (local-only)
//! - **PV creation**: `create_pv_double`, `create_pv_int32`, `create_pv_string`,
//!   `create_pv_enum`, and their `_array` variants
//! - **POST**: Publish new values with automatic alarm computation —
//!   `post_double`, `post_int32`, `post_string`, `post_enum`, and `_array` variants
//! - **Fetch**: Read the current server-side value with alarm info —
//!   `fetch_double`, `fetch_int32`, `fetch_string`, `fetch_enum`
//! - **Stop**: [`Server::stop_drop`] — consumes the server and frees all resources
//! - **Handle**: [`ServerHandle`] — clone-able, thread-safe handle for use across threads
//!
//! ### Metadata & Alarms
//! - [`NTScalarMetadataBuilder`] / [`NTEnumMetadataBuilder`]: configure PV metadata at
//!   creation time (display limits, units, precision, control limits, value alarm thresholds)
//! - [`ControlMetadata`], [`AlarmMetadata`]: structs passed to the builders
//! - [`AlarmSeverity`], [`AlarmStatus`]: alarm state reported in fetched values and monitors
//!
//! ### Other
//! - **Logging**: [`set_logger_level`] — programmatically set PVXS log levels
//! - Thread-safe [`Context`] (implements `Send + Sync`)
//!

pub mod bridge;

mod alarms;
mod client;
mod metadata;
mod server;
mod value;

use std::fmt;

pub use bridge::{
    ContextWrapper, MonitorBuilderWrapper, MonitorWrapper, RpcWrapper, ServerWrapper,
    SharedPVWrapper, StaticSourceWrapper, ValueWrapper,
};

/// Configure PVXS logging from environment variable `PVXS_LOG`
///
/// Reads the `PVXS_LOG` environment variable to configure logging levels.
/// Format: "logger_name=LEVEL,another=LEVEL"
///
/// Examples:
/// - `PVXS_LOG="*=DEBUG"` - all loggers at DEBUG level
/// - `PVXS_LOG="pvxs.*=INFO"` - all internal loggers at INFO
/// - `PVXS_LOG="pvxs.tcp.io=CRIT"` - suppress tcp.io errors below CRIT
///
/// Levels: CRIT < ERR < WARN < INFO < DEBUG
///
/// # Example
///
/// ```no_run
/// use pvxs_sys::configure_logging_from_env;
///
/// // Read from PVXS_LOG environment variable
/// configure_logging_from_env().ok();
/// ```
pub fn configure_logging_from_env() -> Result<()> {
    bridge::pvxs_logger_config_env().map_err(|e| e.into())
}

/// Set logging level for a specific PVXS logger
///
/// Programmatically set the log level for a named logger.
///
/// # Arguments
///
/// * `name` - Logger name (e.g., "pvxs.tcp.io", "pvxs.*" for wildcards)
/// * `level` - One of: "CRIT", "ERR", "WARN", "INFO", "DEBUG"
///
/// # Example
///
/// ```no_run
/// use pvxs_sys::set_logger_level;
///
/// // Suppress benign TCP disconnect errors (socket error 10054)
/// set_logger_level("pvxs.tcp.io", "CRIT").ok();
/// ```
pub fn set_logger_level(name: &str, level: &str) -> Result<()> {
    bridge::pvxs_logger_level_set(name.to_string(), level.to_string()).map_err(|e| e.into())
}

pub use alarms::{compute_alarm_for_scalar, AlarmConfig, AlarmResult, AlarmSeverity, AlarmStatus};
pub use client::{Context, Monitor, MonitorBuilder, MonitorEvent, Rpc};
pub use metadata::{AlarmMetadata, ControlMetadata, DisplayMetadata};
pub use server::{
    FetchedDouble, FetchedDoubleArray, FetchedEnum, FetchedInt32, FetchedInt32Array, FetchedString,
    FetchedStringArray, NTEnumMetadataBuilder, NTScalarMetadataBuilder, Server, ServerHandle,
    SharedPV, StaticSource,
};
pub use value::Value;

// Re-export for testing callbacks
pub use std::sync::atomic::{AtomicUsize, Ordering};

// Re-export for convenience
pub type Result<T> = std::result::Result<T, PvxsError>;

/// Error type for PVXS operations
#[derive(Debug, Clone)]
pub struct PvxsError {
    message: String,
}

impl PvxsError {
    pub fn new(message: impl Into<String>) -> Self {
        Self {
            message: message.into(),
        }
    }
}

impl fmt::Display for PvxsError {
    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
        write!(f, "PVXS error: {}", self.message)
    }
}

impl std::error::Error for PvxsError {}

impl From<cxx::Exception> for PvxsError {
    fn from(e: cxx::Exception) -> Self {
        Self::new(e.what())
    }
}