fennec-modbus 0.93.1

Modular Modbus client implementation
Documentation
//! The lowest protocol level.
//!
//! It operates with PDU's and independent of any transport.
//! If you're implementing transport like PDU, you're going to need this module:
//!
//! - **Data units** are the PDU's that you're going to wrap into your transport.
//! - **Functions** are the actual Modbus functions expressed in terms of function code,
//!   request arguments and output.

pub mod address;
pub mod codec;
pub mod function;

use bytes::{Buf, BufMut};

use crate::{
    Error,
    protocol::{
        codec::{Decode, Encode},
        function::IntoValue,
    },
};

/// Request Protocol Data Unit.
#[derive(Copy, Clone)]
pub struct Request<A> {
    /// Modbus function code.
    pub function_code: u8,

    /// Function-dependent arguments that follow the function code.
    pub args: A,
}

impl<A> Request<A> {
    /// Wrap the function arguments into PDU.
    pub const fn wrap<F: Function<Args = A>>(args: A) -> Self {
        Self { function_code: F::CODE, args }
    }
}

impl<A: Encode> Encode for Request<A> {
    fn encode(&self, to: &mut impl BufMut) {
        to.put_u8(self.function_code);
        self.args.encode(to);
    }
}

/// Response Protocol Data Unit.
#[derive(Copy, Clone)]
pub enum Response<F: Function> {
    /// Successful response.
    Ok(F::Output),

    /// The connection is healthy, but the response is a Modbus exception.
    Exception(Exception),
}

impl<F: Function> Decode for Response<F> {
    fn decode(from: &mut impl Buf) -> Result<Self, Error> {
        match from.try_get_u8()? {
            function_code if function_code == F::CODE => Ok(Self::Ok(F::Output::decode(from)?)),
            function_code if function_code == (F::CODE | 0x80) => {
                Ok(Self::Exception(Exception::decode(from)?))
            }
            function_code => Err(Error::UnexpectedFunctionCode(function_code)),
        }
    }
}

impl<F: Function> Response<F> {
    pub fn into_result(self) -> Result<F::Output, Error> {
        match self {
            Self::Ok(output) => Ok(output),
            Self::Exception(exception) => Err(Error::Exception(exception)),
        }
    }
}

/// High-level protocol error.
///
/// The server received the request without a communication error, but could not handle it.
#[must_use]
#[derive(Copy, Clone, Debug, thiserror::Error)]
pub enum Exception {
    /// The function code received in the query is not an allowable action for the server:
    ///
    /// - the function was not implemented in the unit selected;
    /// - the server is in the wrong state to process a request of this type.
    #[error("illegal function")]
    IllegalFunction,

    /// The data address received in the query is not an allowable address for the server.
    ///
    /// The combination of reference number and transfer length is invalid.
    #[error("illegal data address")]
    IllegalDataAddress,

    /// A value contained in the query data field is not an allowable value for server.
    #[error("illegal data value")]
    IllegalDataValue,

    /// An unrecoverable error occurred while the server was attempting to perform the requested action.
    #[error("server device failure")]
    ServerDeviceFailure,

    /// The server has accepted the request and is processing it, but a long duration of time will be
    /// required to do so.
    ///
    /// This response is returned to prevent a timeout error from occurring in the client.
    /// The client can next issue a «Poll Program Complete» message to determine if processing is completed.
    #[error("acknowledge")]
    Acknowledge,

    /// The server is engaged in processing a long–duration program command.
    ///
    /// The client should retransmit the message later when the server is free.
    #[error("server device busy")]
    ServerDeviceBusy,

    /// The server attempted to read record file, but  detected a parity error in the memory.
    ///
    /// The client can retry the request, but service may be required on the server device.
    #[error("memory parity error")]
    MemoryParityError,

    /// The gateway was unable to allocate an internal communication path from the input port
    /// to the output port for processing the request.
    #[error("gateway path unavailable")]
    GatewayPathUnavailable,

    /// No response was obtained from the target device.
    ///
    /// Usually means that the device is not present on the network.
    #[error("gateway target device failed to respond")]
    GatewayTargetDeviceFailedToRespond,

    /// Non-standard error code.
    #[error("custom error ({0})")]
    Custom(u8),
}

impl Decode for Exception {
    fn decode(from: &mut impl Buf) -> Result<Self, Error> {
        match from.try_get_u8()? {
            0x01 => Ok(Self::IllegalFunction),
            0x02 => Ok(Self::IllegalDataAddress),
            0x03 => Ok(Self::IllegalDataValue),
            0x04 => Ok(Self::ServerDeviceFailure),
            0x05 => Ok(Self::Acknowledge),
            0x06 => Ok(Self::ServerDeviceBusy),
            0x08 => Ok(Self::MemoryParityError),
            0x0A => Ok(Self::GatewayPathUnavailable),
            0x0B => Ok(Self::GatewayTargetDeviceFailedToRespond),
            exception_code => Ok(Self::Custom(exception_code)),
        }
    }
}

/// Trait that ties function code, arguments and output together.
///
/// Users are free to implement their own functions – be that custom Modbus functions
/// or alternate standard function implementations. In the latter case, consider
/// [making a pull request](https://github.com/eigenein/fennec/pulls).
pub trait Function: function::Code {
    /// Function arguments type.
    ///
    /// It must be encodable to get sent in the request.
    type Args: Encode;

    /// Function output type.
    ///
    /// It must be decodable from the response.
    type Output: Decode + IntoValue;
}

/// Marker trait to separate addresses from any other encodable types.
pub trait Address: Encode {}

impl Address for u16 {}