ntex-amqp 5.8.0

AMQP 1.0 Client/Server framework
Documentation
use std::{cell::RefCell, fmt, rc::Rc};

use ntex_bytes::ByteString;
use ntex_router::Path;
use ntex_util::future::Either;

use crate::codec::protocol::{Accepted, Attach, DeliveryState, Detach, Error, Flow, Rejected};
use crate::{
    Handle, State, error::AmqpProtocolError, rcvlink::ReceiverLink, session::Session,
    sndlink::SenderLink,
};

pub use crate::codec::protocol::Transfer;

#[derive(Debug)]
pub enum Message {
    Attached(Attach, ReceiverLink),
    Detached(ReceiverLink),
    DetachedAll(Vec<ReceiverLink>),
    Transfer(ReceiverLink),
}

#[derive(Debug)]
pub(crate) enum Action {
    None,
    AttachSender(SenderLink, Attach, Attach),
    AttachReceiver(ReceiverLink, Attach, Attach),
    DetachSender(SenderLink, Detach),
    DetachReceiver(ReceiverLink, Detach),
    SessionEnded(Vec<Either<SenderLink, ReceiverLink>>),
    Flow(SenderLink, Flow),
    Transfer(ReceiverLink),
    RemoteClose(AmqpProtocolError),
}

pub struct Link<S> {
    pub(crate) state: State<S>,
    pub(crate) link: ReceiverLink,
    pub(crate) path: Path<ByteString>,
    pub(crate) attach: Rc<Attach>,
}

impl<S> Link<S> {
    pub(crate) fn new(
        attach: Attach,
        link: ReceiverLink,
        state: State<S>,
        path: ByteString,
    ) -> Self {
        Link {
            state,
            link,
            attach: Rc::new(attach),
            path: Path::new(path),
        }
    }

    pub fn path(&self) -> &Path<ByteString> {
        &self.path
    }

    pub fn path_mut(&mut self) -> &mut Path<ByteString> {
        &mut self.path
    }

    pub fn state(&self) -> &State<S> {
        &self.state
    }

    pub fn handle(&self) -> Handle {
        self.link.handle()
    }

    pub fn frame(&self) -> &Attach {
        self.attach.as_ref()
    }

    pub fn session(&self) -> &Session {
        self.link.session()
    }

    pub fn receiver(&self) -> &ReceiverLink {
        &self.link
    }

    pub fn receiver_mut(&mut self) -> &mut ReceiverLink {
        &mut self.link
    }

    pub fn link_credit(&self, credit: u32) {
        self.link.set_link_credit(credit);
    }
}

impl<S> Clone for Link<S> {
    fn clone(&self) -> Self {
        Self {
            state: self.state.clone(),
            link: self.link.clone(),
            path: self.path.clone(),
            attach: self.attach.clone(),
        }
    }
}

impl<S> fmt::Debug for Link<S> {
    fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
        fmt.debug_struct("Link<S>")
            .field("frame", self.frame())
            .finish()
    }
}

#[derive(Debug, From)]
pub enum Outcome {
    Accept,
    Reject,
    Error(Error),
}

impl Outcome {
    pub(crate) fn into_delivery_state(self) -> DeliveryState {
        match self {
            Outcome::Accept => DeliveryState::Accepted(Accepted {}),
            Outcome::Reject => DeliveryState::Rejected(Rejected { error: None }),
            Outcome::Error(e) => DeliveryState::Rejected(Rejected { error: Some(e) }),
        }
    }
}

#[derive(Debug)]
pub struct Wrapper<T> {
    inner: RefCell<Option<T>>,
}

impl<T> Wrapper<T> {
    pub(crate) fn new(inner: T) -> Self {
        Self {
            inner: RefCell::new(Some(inner)),
        }
    }

    pub(crate) fn take(&self) -> T {
        self.inner.borrow_mut().take().unwrap()
    }

    #[allow(clippy::missing_panics_doc)]
    pub fn with<F, R>(&self, f: F) -> R
    where
        F: FnOnce(&mut T) -> R,
    {
        f(self.inner.borrow_mut().as_mut().unwrap())
    }
}