diff --git a/src/service/mod.rs b/src/service/mod.rs index fb36240..b3ba92e 100644 --- a/src/service/mod.rs +++ b/src/service/mod.rs @@ -7,6 +7,26 @@ pub(crate) mod rtu; #[cfg(feature = "tcp")] pub(crate) mod tcp; +#[cfg(any(feature = "rtu", feature = "tcp"))] +async fn disconnect(framed: tokio_util::codec::Framed) -> std::io::Result<()> +where + T: tokio::io::AsyncWrite + Unpin, +{ + use tokio::io::AsyncWriteExt as _; + + framed + .into_inner() + .shutdown() + .await + .or_else(|err| match err.kind() { + std::io::ErrorKind::NotConnected | std::io::ErrorKind::BrokenPipe => { + // Already disconnected. + Ok(()) + } + _ => Err(err), + }) +} + /// Check that `req_hdr` is the same `Header` as `rsp_hdr`. /// /// # Errors diff --git a/src/service/rtu.rs b/src/service/rtu.rs index 41a7016..0f4be17 100644 --- a/src/service/rtu.rs +++ b/src/service/rtu.rs @@ -4,7 +4,7 @@ use std::{fmt, io}; use futures_util::{SinkExt as _, StreamExt as _}; -use tokio::io::{AsyncRead, AsyncWrite, AsyncWriteExt as _}; +use tokio::io::{AsyncRead, AsyncWrite}; use tokio_util::codec::Framed; use crate::{ @@ -14,7 +14,7 @@ use crate::{ ProtocolError, Result, }; -use super::verify_response_header; +use super::{disconnect, verify_response_header}; /// Modbus RTU client #[derive(Debug)] @@ -106,17 +106,7 @@ where // Already disconnected. return Ok(()); }; - framed - .into_inner() - .shutdown() - .await - .or_else(|err| match err.kind() { - io::ErrorKind::NotConnected | io::ErrorKind::BrokenPipe => { - // Already disconnected. - Ok(()) - } - _ => Err(err), - }) + disconnect(framed).await } } diff --git a/src/service/tcp.rs b/src/service/tcp.rs index 0e2d08b..0aea309 100644 --- a/src/service/tcp.rs +++ b/src/service/tcp.rs @@ -4,7 +4,7 @@ use std::{fmt, io}; use futures_util::{SinkExt as _, StreamExt as _}; -use tokio::io::{AsyncRead, AsyncWrite, AsyncWriteExt as _}; +use tokio::io::{AsyncRead, AsyncWrite}; use tokio_util::codec::Framed; use crate::{ @@ -18,6 +18,8 @@ use crate::{ ExceptionResponse, ProtocolError, Request, Response, Result, }; +use super::disconnect; + const INITIAL_TRANSACTION_ID: TransactionId = 0; #[derive(Debug)] @@ -137,17 +139,7 @@ where // Already disconnected. return Ok(()); }; - framed - .into_inner() - .shutdown() - .await - .or_else(|err| match err.kind() { - io::ErrorKind::NotConnected | io::ErrorKind::BrokenPipe => { - // Already disconnected. - Ok(()) - } - _ => Err(err), - }) + disconnect(framed).await } }