From d328e6c2f776e0934b3ec45edef46cb787abf4e7 Mon Sep 17 00:00:00 2001
From: Christian Visintin
Date: Mon, 14 Oct 2024 14:59:50 +0200
Subject: [PATCH] fix: Passive mode with custom provided TcpStream (#91)
* fix: Passive mode with custom provided TcpStream
* fix: ci
* fix: ci
---
.github/workflows/cargo.yml | 41 ++++--------------
.github/workflows/cli.yml | 7 +---
.github/workflows/coverage.yml | 9 ++--
CHANGELOG.md | 7 ++++
Cargo.toml | 2 +-
README.md | 2 +-
suppaftp/Cargo.toml | 2 +-
suppaftp/src/async_ftp/mod.rs | 59 +++++++++++++++++++++++---
suppaftp/src/sync_ftp/mod.rs | 64 ++++++++++++++++++++++-------
suppaftp/src/sync_ftp/tls/rustls.rs | 7 ++--
10 files changed, 131 insertions(+), 69 deletions(-)
diff --git a/.github/workflows/cargo.yml b/.github/workflows/cargo.yml
index ca39ff5..278eb57 100644
--- a/.github/workflows/cargo.yml
+++ b/.github/workflows/cargo.yml
@@ -10,50 +10,27 @@ jobs:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
- - uses: actions-rs/toolchain@v1
+ - uses: dtolnay/rust-toolchain@stable
with:
toolchain: stable
override: true
components: rustfmt, clippy
- name: Setup containers
- run: docker-compose -f "tests/docker-compose.yml" up -d --build
+ run: docker compose -f "tests/docker-compose.yml" up -d --build
- name: Build simple
- uses: actions-rs/cargo@v1
- with:
- command: build
- args: --package suppaftp
+ run: cargo build --package suppaftp
- name: Build secure (native-tls)
- uses: actions-rs/cargo@v1
- with:
- command: build
- args: --features native-tls,deprecated --package suppaftp
+ run: cargo build --features native-tls,deprecated --package suppaftp
- name: Build secure (rustls)
- uses: actions-rs/cargo@v1
- with:
- command: build
- args: --features rustls,deprecated --package suppaftp
+ run: cargo build --features rustls,deprecated --package suppaftp
- name: Build async
- uses: actions-rs/cargo@v1
- with:
- command: build
- args: --features async,deprecated --package suppaftp
+ run: cargo build --features async,deprecated --package suppaftp
- name: Build async-native-tls
- uses: actions-rs/cargo@v1
- with:
- command: build
- args: --features async-native-tls,deprecated --package suppaftp
+ run: cargo build --features async-native-tls,deprecated --package suppaftp
- name: Build all features
- uses: actions-rs/cargo@v1
- with:
- command: build
- args: --features deprecated,native-tls,rustls,async-native-tls,async-rustls --package suppaftp
+ run: cargo build --features deprecated,native-tls,rustls,async-native-tls,async-rustls --package suppaftp
- name: Run tests
- uses: actions-rs/cargo@v1
- with:
- command: test
- args: --lib --package suppaftp --no-default-features --features rustls,native-tls,async-native-tls,async-rustls,with-containers --no-fail-fast
- env:
- RUST_LOG: trace
+ run: cargo test --lib --package suppaftp --no-default-features --features rustls,native-tls,async-native-tls,async-rustls,with-containers --no-fail-fast
- name: Format
run: cargo fmt --all -- --check
- name: Clippy
diff --git a/.github/workflows/cli.yml b/.github/workflows/cli.yml
index 471f77c..a6c9718 100644
--- a/.github/workflows/cli.yml
+++ b/.github/workflows/cli.yml
@@ -13,16 +13,13 @@ jobs:
working-directory: ./suppaftp-cli
steps:
- uses: actions/checkout@v2
- - uses: actions-rs/toolchain@v1
+ - uses: dtolnay/rust-toolchain@stable
with:
toolchain: stable
override: true
components: rustfmt, clippy
- name: Build
- uses: actions-rs/cargo@v1
- with:
- command: build
- args: --package suppaftp-cli
+ run: cargo build --package suppaftp-cli
- name: Format
run: cargo fmt --all -- --check
- name: Clippy
diff --git a/.github/workflows/coverage.yml b/.github/workflows/coverage.yml
index 7cf1ffc..d0501b8 100644
--- a/.github/workflows/coverage.yml
+++ b/.github/workflows/coverage.yml
@@ -12,17 +12,14 @@ jobs:
steps:
- uses: actions/checkout@v2
- name: Setup containers
- run: docker-compose -f "tests/docker-compose.yml" up -d --build
+ run: docker compose -f "tests/docker-compose.yml" up -d --build
- name: Setup nightly toolchain
- uses: actions-rs/toolchain@v1
+ uses: dtolnay/rust-toolchain@stable
with:
toolchain: nightly
override: true
- name: Run tests (nightly)
- uses: actions-rs/cargo@v1
- with:
- command: test
- args: --lib --no-default-features --features native-tls,deprecated,async-native-tls,with-containers --no-fail-fast
+ run: cargo test --lib --package suppaftp --no-default-features --features native-tls,deprecated,async-native-tls,with-containers --no-fail-fast
env:
RUST_LOG: trace
CARGO_INCREMENTAL: "0"
diff --git a/CHANGELOG.md b/CHANGELOG.md
index c36e95c..ce83279 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,6 +1,7 @@
# Changelog
- [Changelog](#changelog)
+ - [6.0.2](#602)
- [6.0.1](#601)
- [6.0.0](#600)
- [5.4.0](#540)
@@ -34,6 +35,12 @@
---
+## 6.0.2
+
+Released on 14/10/2024
+
+- [Issue 89](https://github.com/veeso/suppaftp/issues/89): added new `FtpStream::passive_stream_builder` to provide a function to build the Passive mode `TcpStream` with a custom builder. This is useful if you need to use some proxy.
+
## 6.0.1
Released on 24/05/2024
diff --git a/Cargo.toml b/Cargo.toml
index 316a348..54739b6 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -3,7 +3,7 @@ members = ["suppaftp", "suppaftp-cli"]
resolver = "2"
[workspace.package]
-version = "6.0.1"
+version = "6.0.2"
edition = "2021"
authors = [
"Christian Visintin ",
diff --git a/README.md b/README.md
index 0c0d1cb..ec8b883 100644
--- a/README.md
+++ b/README.md
@@ -11,7 +11,7 @@
Developed by veeso and Matt McCoy
-Current version: 6.0.1 (24/05/2024)
+Current version: 6.0.2 (14/10/2024)
Pin> + Send>>;
+
/// Stream to interface with the FTP server. This interface is only for the command stream.
pub struct ImplAsyncFtpStream
where
@@ -45,6 +53,7 @@ where
nat_workaround: bool,
welcome_msg: Option,
active_timeout: Duration,
+ passive_stream_builder: Box,
#[cfg(not(feature = "async-secure"))]
marker: PhantomData,
#[cfg(feature = "async-secure")]
@@ -85,6 +94,7 @@ where
marker: PhantomData {},
mode: Mode::Passive,
nat_workaround: false,
+ passive_stream_builder: Self::default_passive_stream_builder(),
welcome_msg: None,
#[cfg(feature = "async-secure")]
tls_ctx: None,
@@ -143,6 +153,7 @@ where
reader: BufReader::new(DataStream::Ssl(Box::new(stream))),
mode: self.mode,
nat_workaround: self.nat_workaround,
+ passive_stream_builder: self.passive_stream_builder,
tls_ctx: Some(Box::new(tls_connector)),
domain: Some(String::from(domain)),
welcome_msg: self.welcome_msg,
@@ -197,6 +208,7 @@ where
mode: Mode::Passive,
nat_workaround: false,
welcome_msg: None,
+ passive_stream_builder: Self::default_passive_stream_builder(),
tls_ctx: None,
domain: None,
active_timeout: Duration::from_secs(60),
@@ -213,6 +225,7 @@ where
reader: BufReader::new(DataStream::Ssl(stream.into())),
mode: Mode::Passive,
nat_workaround: false,
+ passive_stream_builder: Self::default_passive_stream_builder(),
tls_ctx: Some(Box::new(tls_connector)),
domain: Some(String::from(domain)),
welcome_msg: None,
@@ -238,6 +251,18 @@ where
self
}
+ /// Set a custom [`StreamBuilder`] for passive mode.
+ ///
+ /// The stream builder is a function that takes a `SocketAddr` and returns a `TcpStream` and it's used
+ /// to create the [`TcpStream`] for the data connection in passive mode.
+ pub fn passive_stream_builder(mut self, stream_builder: F) -> Self
+ where
+ F: Fn(SocketAddr) -> Pin> + Send>> + 'static,
+ {
+ self.passive_stream_builder = Box::new(stream_builder);
+ self
+ }
+
/// Returns welcome message retrieved from server (if available)
pub fn get_welcome_msg(&self) -> Option<&str> {
self.welcome_msg.as_deref()
@@ -767,16 +792,12 @@ where
Mode::ExtendedPassive => {
let addr = self.epsv().await?;
self.perform(cmd).await?;
- TcpStream::connect(addr)
- .await
- .map_err(FtpError::ConnectionError)?
+ (self.passive_stream_builder)(addr).await?
}
Mode::Passive => {
let addr = self.pasv().await?;
self.perform(cmd).await?;
- TcpStream::connect(addr)
- .await
- .map_err(FtpError::ConnectionError)?
+ (self.passive_stream_builder)(addr).await?
}
};
@@ -1002,6 +1023,16 @@ where
self.finalize_retr_stream(data_stream).await?;
lines
}
+
+ fn default_passive_stream_builder() -> Box {
+ Box::new(|address| {
+ Box::pin(async move {
+ TcpStream::connect(address)
+ .await
+ .map_err(FtpError::ConnectionError)
+ })
+ })
+ }
}
#[cfg(test)]
@@ -1400,6 +1431,22 @@ mod test {
finalize_stream(stream).await;
}
+ #[async_attributes::test]
+ async fn test_should_set_passive_stream_builder() {
+ crate::log_init();
+ let _ftp_stream = AsyncFtpStream::connect("test.rebex.net:21")
+ .await
+ .unwrap()
+ .passive_stream_builder(|addr| {
+ Box::pin(async move {
+ println!("Connecting to {}", addr);
+ TcpStream::connect(addr)
+ .await
+ .map_err(FtpError::ConnectionError)
+ })
+ });
+ }
+
// -- test utils
#[cfg(feature = "with-containers")]
diff --git a/suppaftp/src/sync_ftp/mod.rs b/suppaftp/src/sync_ftp/mod.rs
index 8dca29b..e5ba5fb 100644
--- a/suppaftp/src/sync_ftp/mod.rs
+++ b/suppaftp/src/sync_ftp/mod.rs
@@ -5,7 +5,6 @@
mod data_stream;
mod tls;
-use std::fmt::Debug;
use std::io::{copy, BufRead, BufReader, Cursor, Read, Write};
#[cfg(not(feature = "secure"))]
use std::marker::PhantomData;
@@ -32,8 +31,12 @@ use crate::command::Command;
use crate::command::ProtectionLevel;
use crate::types::Features;
+/// A function that creates a new stream for the data connection in passive mode.
+///
+/// It takes a [`SocketAddr`] and returns a [`TcpStream`].
+pub type PassiveStreamBuilder = dyn Fn(SocketAddr) -> FtpResult + Send;
+
/// Stream to interface with the FTP server. This interface is only for the command stream.
-#[derive(Debug)]
pub struct ImplFtpStream
where
T: TlsStream,
@@ -43,6 +46,7 @@ where
nat_workaround: bool,
welcome_msg: Option,
active_timeout: Duration,
+ passive_stream_builder: Box,
#[cfg(not(feature = "secure"))]
marker: PhantomData,
#[cfg(feature = "secure")]
@@ -80,6 +84,7 @@ where
nat_workaround: false,
welcome_msg: None,
active_timeout: Duration::from_secs(60),
+ passive_stream_builder: Self::default_passive_stream_builder(),
#[cfg(feature = "secure")]
tls_ctx: None,
#[cfg(feature = "secure")]
@@ -106,6 +111,18 @@ where
self
}
+ /// Set a custom [`StreamBuilder`] for passive mode.
+ ///
+ /// The stream builder is a function that takes a `SocketAddr` and returns a `TcpStream` and it's used
+ /// to create the [`TcpStream`] for the data connection in passive mode.
+ pub fn passive_stream_builder(mut self, stream_builder: F) -> Self
+ where
+ F: Fn(SocketAddr) -> FtpResult + Send + 'static,
+ {
+ self.passive_stream_builder = Box::new(stream_builder);
+ self
+ }
+
/// Set the data channel transfer mode
pub fn set_mode(&mut self, mode: Mode) {
debug!("Changed mode to {:?}", mode);
@@ -153,6 +170,7 @@ where
reader: BufReader::new(DataStream::Ssl(Box::new(stream))),
mode: self.mode,
nat_workaround: self.nat_workaround,
+ passive_stream_builder: self.passive_stream_builder,
tls_ctx: Some(Box::new(tls_connector)),
domain: Some(String::from(domain)),
welcome_msg: self.welcome_msg,
@@ -169,7 +187,7 @@ where
/// Connect to remote ftps server using IMPLICIT secure connection.
///
- /// > Warning: mind that implicit ftps should be considered deprecated, if you can use explicit mode with `into_secure()`
+ /// > Warning: mind that implicit ftps should be considered deprecated, if you can use explicit mode with [`ImplFtpStream::into_secure`]
///
///
/// ## Example
@@ -179,8 +197,8 @@ where
/// use suppaftp::native_tls::{TlsConnector, TlsStream};
/// use std::path::Path;
///
- /// // Create a TlsConnector
- /// // NOTE: For custom options see
+ /// //Create a TlsConnector
+ /// //NOTE: For custom options see
/// let mut ctx = TlsConnector::new().unwrap();
/// let mut ftp_stream = FtpStream::connect_secure_implicit("127.0.0.1:990", ctx, "localhost").unwrap();
/// ```
@@ -200,6 +218,7 @@ where
reader: BufReader::new(DataStream::Tcp(stream)),
mode: Mode::Passive,
nat_workaround: false,
+ passive_stream_builder: Self::default_passive_stream_builder(),
welcome_msg: None,
tls_ctx: None,
domain: None,
@@ -217,6 +236,7 @@ where
mode: Mode::Passive,
nat_workaround: false,
tls_ctx: Some(Box::new(tls_connector)),
+ passive_stream_builder: Self::default_passive_stream_builder(),
domain: Some(String::from(domain)),
welcome_msg: None,
active_timeout: Duration::from_secs(60),
@@ -239,7 +259,7 @@ where
self.welcome_msg.as_deref()
}
- /// Returns a reference to the underlying TcpStream.
+ /// Returns a reference to the underlying [`TcpStream`].
///
/// Example:
/// ```no_run
@@ -439,7 +459,7 @@ where
/// This method is a more complicated way to retrieve a file.
/// The reader returned should be dropped.
/// Also you will have to read the response to make sure it has the correct value.
- /// Once file has been read, call `finalize_retr_stream()`
+ /// Once file has been read, call [`ImplFtpStream::finalize_retr_stream`]
pub fn retr_as_stream>(&mut self, file_name: S) -> FtpResult> {
debug!("Retrieving '{}'", file_name.as_ref());
let data_stream = self.data_command(Command::Retr(file_name.as_ref().to_string()))?;
@@ -447,7 +467,7 @@ where
Ok(data_stream)
}
- /// Finalize retr stream; must be called once the requested file, got previously with `retr_as_stream()` has been read
+ /// Finalize retr stream; must be called once the requested file, got previously with [`ImplFtpStream::retr_as_stream`] has been read
pub fn finalize_retr_stream(&mut self, stream: impl Read) -> FtpResult<()> {
debug!("Finalizing retr stream");
// Drop stream NOTE: must be done first, otherwise server won't return any response
@@ -475,7 +495,7 @@ where
}
/// This stores a file on the server.
- /// r argument must be any struct which implemenents the Read trait.
+ /// r argument must be any struct which implemenents the [`Read`] trait.
/// Returns amount of written bytes
pub fn put_file, R: Read>(&mut self, filename: S, r: &mut R) -> FtpResult {
// Get stream
@@ -488,7 +508,7 @@ where
/// Send PUT command and returns a BufWriter, which references the file created on the server
/// The returned stream must be then correctly manipulated to write the content of the source file to the remote destination
/// The stream must be then correctly dropped.
- /// Once you've finished the write, YOU MUST CALL THIS METHOD: `finalize_put_stream`
+ /// Once you've finished the write, YOU MUST CALL THIS METHOD: [`ImplFtpStream::finalize_put_stream`]
pub fn put_with_stream>(&mut self, filename: S) -> FtpResult> {
debug!("Put file {}", filename.as_ref());
let stream = self.data_command(Command::Store(filename.as_ref().to_string()))?;
@@ -498,7 +518,7 @@ where
/// Finalize put when using stream
/// This method must be called once the file has been written and
- /// `put_with_stream` has been used to write the file
+ /// [`ImplFtpStream::put_with_stream`] has been used to write the file
pub fn finalize_put_stream(&mut self, stream: impl Write) -> FtpResult<()> {
debug!("Finalizing put stream");
// Drop stream NOTE: must be done first, otherwise server won't return any response
@@ -510,7 +530,7 @@ where
}
/// Open specified file for appending data. Returns the stream to append data to specified file.
- /// Once you've finished the write, YOU MUST CALL THIS METHOD: `finalize_put_stream`
+ /// Once you've finished the write, YOU MUST CALL THIS METHOD: [`ImplFtpStream::finalize_put_stream`]
pub fn append_with_stream>(&mut self, filename: S) -> FtpResult> {
debug!("Appending to file {}", filename.as_ref());
let stream = self.data_command(Command::Appe(filename.as_ref().to_string()))?;
@@ -872,11 +892,11 @@ where
Mode::ExtendedPassive => self
.epsv()
.and_then(|addr| self.perform(cmd).map(|_| addr))
- .and_then(|addr| TcpStream::connect(addr).map_err(FtpError::ConnectionError))?,
+ .and_then(|addr| (self.passive_stream_builder)(addr))?,
Mode::Passive => self
.pasv()
.and_then(|addr| self.perform(cmd).map(|_| addr))
- .and_then(|addr| TcpStream::connect(addr).map_err(FtpError::ConnectionError))?,
+ .and_then(|addr| (self.passive_stream_builder)(addr))?,
};
#[cfg(not(feature = "secure"))]
@@ -992,6 +1012,11 @@ where
self.finalize_retr_stream(data_stream)?;
lines
}
+
+ /// Default stream builder
+ fn default_passive_stream_builder() -> Box {
+ Box::new(|addr| TcpStream::connect(addr).map_err(FtpError::ConnectionError))
+ }
}
#[cfg(test)]
@@ -1461,4 +1486,15 @@ mod test {
.with_no_client_auth()
}
*/
+
+ #[test]
+ fn test_should_set_passive_stream_builder() {
+ crate::log_init();
+ let _ftp_stream = FtpStream::connect("test.rebex.net:21")
+ .unwrap()
+ .passive_stream_builder(|addr| {
+ println!("Connecting to {}", addr);
+ TcpStream::connect(addr).map_err(FtpError::ConnectionError)
+ });
+ }
}
diff --git a/suppaftp/src/sync_ftp/tls/rustls.rs b/suppaftp/src/sync_ftp/tls/rustls.rs
index f630371..b21319f 100644
--- a/suppaftp/src/sync_ftp/tls/rustls.rs
+++ b/suppaftp/src/sync_ftp/tls/rustls.rs
@@ -6,7 +6,8 @@ use std::io::Write;
use std::net::TcpStream;
use std::sync::Arc;
-use rustls::{ClientConfig, ClientConnection, ServerName, StreamOwned};
+use rustls::pki_types::ServerName;
+use rustls::{ClientConfig, ClientConnection, StreamOwned};
use super::{TlsConnector, TlsStream};
use crate::{FtpError, FtpResult};
@@ -32,8 +33,8 @@ impl TlsConnector for RustlsConnector {
type Stream = RustlsStream;
fn connect(&self, domain: &str, stream: TcpStream) -> FtpResult {
- let server_name =
- ServerName::try_from(domain).map_err(|e| FtpError::SecureError(e.to_string()))?;
+ let server_name = ServerName::try_from(domain.to_string())
+ .map_err(|e| FtpError::SecureError(e.to_string()))?;
let connection = ClientConnection::new(Arc::clone(&self.connector), server_name)
.map_err(|e| FtpError::SecureError(e.to_string()))?;
let stream = StreamOwned::new(connection, stream);