- New: Detect Titan FTP servers (allows for future server-specific handling)
- Fix: Validation of short CRC checksum fails due to mismatch of hex hash format
- Change: Remove redundant and extranous
OpenRead
/OpenWrite
/OpenAppend
API and keep only 2 methods each - Change: Mark
OpenRead
/OpenWrite
/OpenAppend
API as obsolete with warnings and recommend high level API - Change: Cleanup dependencies for netstandard2.0, netstandard2.1 and net5.0 targets (thanks jnyrup)
- Change: Restore the older
OpenRead
API to prevent breaking older projects that depend on it (thanks FanDjango)
- New:
GetZOSFileSize
is now removed and superceeded byGetFileSize
which handles z/OS servers - New: Refactor and cleanup z/OS specific logic for: post-connect init,
IsRoot
,GetFileSize
- Change:
OpenRead
API no longer supportscheckIfFileExists
argument (thanks FanDjango) - Fix: z/OS & ASCII transfers: Don't get filesize if filesize already known during downloading (thanks FanDjango)
- Fix: z/OS
UploadFilesAsync
: Fixed IBM MVS File uploading path calculation (thanks FanDjango) - Fix: Close underlying FTP socket connection on async cancellation on .NET Core (thanks datvm)
- Fix: Correctly handle FTP connection timeout on .NET Core (thanks datvm)
- Fix: Enable detection of IBM OS/400 servers that were disabled during server-specific handling
- New: Detect PyFtpdLib FTP servers (allows for future server-specific handling)
- Fix: Pass CancellationToken to all methods that support cancellation (thanks 0xced)
- Fix: Error check on z/OS init commands to ensure they executed correctly (thanks FanDjango)
- Fix: Improve
FileExists
for z/OS: better no SIZE, no MDTM on non HFS files (thanks FanDjango) - Fix:
DownloadFile
for z/OS:SetDataType
directly before theRETR
command (thanks FanDjango) - Fix:
IsAuthenticated
is not updated when callingConnectAsync
(thanks datvm) - Fix: Reduce number of times
SetDataType
is called internally to improve performance (thanks FanDjango) - Fix: Fail to detect z/OS server if
Unix
is also mentioned in welcome message (thanks FanDjango)
- Fix: z/OS GetFileSize: Ignore SIZE capability even if advertised by server as pointless (thanks FanDjango)
- Fix: z/OS DownloadFile: Read to end of stream because filesize is always inaccurate (thanks FanDjango)
- Fix: z/OS DownloadFile: Fix check for infinity or NaN progress values (thanks FanDjango)
- Fix: z/OS GetListing: Path can be null causing an exception (thanks FanDjango)
- Fix: z/OS GetListing: Large files overflow on size calculation resulting in negative file sizes (thanks FanDjango)
- Fix: z/OS GetListing: Listing fails unless users
CWD
to the correct folder of a non-RECFM=U
PDS (thanks FanDjango)
- New: Support for connecting to FTP/FTPS servers via a SOCKS5 proxy (thanks bjth)
- New: Autoconfigure IBM z/OS FTP server using
SITE DATASETMODE
andQUOTESOVERRIDE
(thanks FanDjango) - Fix:
GetListing
itemFullname
is now correctly calculated for Unix and z/OS systems (thanks FanDjango)
- New:
GetZOSFileSize
APIs to get file size of IBM z/OS file system objects (thanks FanDjango) - New:
GetZOSListRealm
APIs to get realm of IBM z/OS servers (thanks FanDjango) - New: Enhance the z/OS listing parser to get LRECL (via XDSS) on behalf of user (thanks FanDjango)
- Fix:
AutoConnect
detects rejected certificates on connection and raisesFtpInvalidCertificateException
(thanks FanDjango) - Fix:
FtpListOption.ForceList
is not being honored by GetListing and machine listings are used instead (thanks FanDjango) - Fix:
GetListing
regression causing many untrue parse fail warnings (thanks FanDjango) - Fix:
GetObjectInfo
is overwriting Modified date of aFtpListItem
if the consecutive MDTM command fails (thanks Dylan-DutchAndBold) - Fix: reusing same FtpClient should reload server capabilities unless its a cloned connection (thanks FanDjango)
- Fix: Executing
CWD
usingExecute
API does not invalidate internal CWD cache (thanks FanDjango)
- Fix:
UploadFile
fails to upload inFtpRemoteExists.Resume
mode even if stream is seekable
- Fix:
AutoConnect
loads the newly detectedFtpProfile
to update properties & encryption - Fix: Passive connections work in
FtpEncryptionMode.Auto
mode and FTPS connection fails
- Fix:
AutoDetect
correctly recommendsFtpEncryptionMode.None
if FTPS connection failed - Fix:
AutoDetect
crashes because attempting to read socket type after disconnected
- Major:
AutoDetect
does not cycle through data connection types during connection as it is irrelevant - Major:
AutoDetect
calculates a data connection type after connection succeeds (EPSV or PASV)
- Fix:
ConnectAsync
now correctly creates a FTP server-specific handler to matchConnect
behaviour
-
Automatic connection
- Major:
AutoConnect
takes far fewer connection attempts due to improvements in connection handling - Major:
AutoConnect
andAutoDetect
are much faster and smarter and only try each setting once if possible - Major:
AutoDetect
only tries Explicit and Implicit FTPS once and then falls back to plaintext FTP - Major:
AutoDetect
only tries UTF-8 and never ASCII because most UTF-8 servers don't advertise it - New:
AutoDetect
verifies if the server supports UTF-8 and updates theFtpProfile
accordingly - New:
FtpProfile
code generation adds a warning message if the encoding mode is unverified - New:
AutoConnectAsync
now uses true asynchronous connection backed by newAutoDetectAsync
- New:
AutoConnect
now auto computes an FTP port unless a non-standard port is already set - New:
AutoConnect
uses the mainFtpClient
connection rather than creating one clone per attempt - Fix:
AutoConnect
remains connected to the first working profile rather than connecting twice on success - Fix:
AutoConnect
reuses the same connection for FTPS and FTP rather than connecting again - Fix: Ensure FTP server capabilities are loaded during
AutoDetect
if original connection is blank - Fix:
AutoConnect
andAutoDetect
will now throw exceptions for permanent failures (bad host/credentials) - Fix:
ConnectAsync
now correctly resets the state flags to matchConnect
behaviour - Fix:
Port
now correctly calculates the default port 21 when usingFtpEncryptionMode.Auto
- Major:
-
Appending and resuming uploads
- Major: The setting
FtpLocalExists.Append
is now renamed toFtpLocalExists.Resume
- Major: The setting
FtpRemoteExists.Append
is now renamed toFtpRemoteExists.Resume
- Major: Split
FtpRemoteExists.Append
into two properties with distinct behaviour (Resume
andAddToEnd
) - Major: Improvements to
UploadFile
andUploadFileAsync
to support appending and resuming of uploads - Major:
UploadFile
always sets the length of the remote file stream before uploading, appending or resuming - Major:
UploadFile
skips uploading inResume
mode if local and remote file are equal length - Fix: Implementation for resuming uploads using
UploadFile
based on fixes inUploadFileAsync
- Major: The setting
-
Machine listings
- Major:
GetListing
prefers using Machine Listings over LIST command, unless a custom list parser is set - Fix:
ListingParser
property is updated according to auto-detected parser duringConnect
andConnectAsync
- Fix:
DeleteDirectory
andDereferenceLink
methods no longer useForceList
and so prefer using Machine Listings
- Major:
-
File hashing
- Major: All low-level hash methods are now inaccessible and
GetChecksum
is the only recommended approach - Fix:
GetChecksum
now prints function call logs and sanitizes the input path - New:
GetChecksum
switches to the first preferred hash algorithm forHASH
command if no algorithm is specified - New:
GetChecksum
validates if the required algorithm is unsupported and throwsFtpHashUnsupportedException
- New:
GetChecksum
validates if hashing is unsupported by the server and throwsFtpHashUnsupportedException
- Fix:
GetChecksumAsync
now takes the cancellation token last to follow conventions (argument reorder) - Fix: Improved extraction of hash checksum when using the HASH command
- Fix: Improved extraction of hash checksum when using the MD5, SHA1, SHA256, SHA512 or X-series commands
- New:
SetHashAlgorithm
now only modifies the hash algorithm if it has changed
- Major: All low-level hash methods are now inaccessible and
-
Path sanitization
- Fix: All high level API methods sanitize input paths to improve robustness
- Fix:
GetWorkingDirectory
always sanitizes the returned working path directory - Fix: Correctly handle server-specific absolute FTP paths for async operations
- Fix: All function call logs now print the sanitized path rather than raw input path
-
Path improvements
- Major:
GetWorkingDirectory
is now extremely fast and caches the working dir path for subsequent calls - Fix:
FileExists
supports checking name listings for Windows NT servers which use invalid slashes - Fix: Root directory FTP paths no longer return
./
and instead return/
- Major:
-
Other improvements
- Major: All legacy asynchronous methods using
IAsyncResult
pattern have been removed (outdated since 2012) - Fix: FXP file transfers for glFTPd server always try PASV and CPSV commands to get passive port
- Fix: Add logging for skipped files in
UploadFile
- Fix: Add file path details in skipped files logged by
UploadFile
andDownloadFile
- New:
GetNameListing
to print results of name listing as verbose logs, similar toGetListing
- New:
GetFileSize
andGetFileSizeAsync
to support a configurable return value if the file does not exist - New:
FtpFolderNameRule
now supportsstartSegment
to skip checking root directory folder names - New:
FtpFolderNameRegexRule
now supportsstartSegment
to skip checking root directory folder names
- Major: All legacy asynchronous methods using
- New: Add support for
IsAuthenticated
property which enables detection of FTP connection and authentication - Fix: Improved file path calculation when uploading files to IBM zOS running MVS (thanks arafuls)
- Fix: Detection of file exists for Windows NT Servers fails due to invalid slashes
- Fix: Ensure
CompletionCode
is always null-checked to fix edge cases and comply with existing implementation - Fix: Possible wrong IP when connecting to Azure FTP Server using
EPSV
command (thanks jsantos74)
- Fix file path calculation when uploading files to IBM zOS running MVS (thanks arafuls)
- Detect fully-qualified directory when using data sets on IBM zOS running MVS (thanks arafuls)
- Major: Refactor and cleanup helpers & extension methods
- Major: Changed namespace of
FtpTrace
toFluentFTP.Helpers
- Major: Changed namespace of
FtpListParser
toFluentFTP.Helpers
- Major: You need to import
FluentFTP.Helpers
to gain access to FTP extension methods - Major: Changed visibility of various internal FTP helpers to
internal
to prevent external access - New: Ability to set a local IP address to be used for FTP connections (thanks daviddenis-stx)
- New: Properties for local IP:
SocketLocalIp
,SocketLocalEndPoint
,SocketRemoteEndPoint
(thanks daviddenis-stx)
- New: Ability to block certain server ports from being used during passive FTP connection (PASV/EPSV)
- New:
PassiveBlockedPorts
andPassiveMaxAttempts
properties to configure passive blocked ports
- Fix: Add support for another server specific string to detect if file exists
- Fix: Prevent memory leaks with FTPS on AKS by disabling
ValidateCertificateRevocation
by default
- Fix:
DownloadFile
should fail if directory path is passed inlocalPath
- Fix:
LastReply
is not set when using async methods - Fix: Ambiguous call when using LINQ extension methods and EF core
- Fix: Remove dependency on
System.Linq.Async
and disableGetListingAsyncEnumerable
- Fix: Use .NET Core API for .NET 5 builds to fix various networking and connectivity issues
- New: Support detection of FRITZ!Box FTP servers (allows for server specific commands)
- Fix: Do not force users to
Connect
if capabilities are not loaded (only force ifCapabilities
are read before connecting)
- Fix: Conditionally hard-abort
AutoConnect
only if the credentials were incorrect
- New:
LogToFile
andLogToConsole
are now available on .NET Framework / .NET 5.0 - Fix:
Capabilities
&HashAlgorithms
are returned if they are loaded regardless of connection status
- Fix: Prevent
Capabilities
&HashAlgorithms
from causing sync-over-async path - Fix:
NullReferenceException
duringUploadFile
if the error message is unknown - Fix: If incorrect credentials are passed to
AutoConnect
, it does not hard abort - Fix: Initializing TLS Authentication hangs in .NET 5 if
SslBuffering
is enabled
- New: Support for .NET 5.0 platform
- New: Support for
IAsyncEnumerable
pattern in .NET Standard 2.0+ and .NET 5.0 (thanks hez2010) - New: Async variant of
GetListingAsync
introduced:GetListingAsyncEnumerable
- Fix: URI ports not being respected when Uri constructor used (thanks julian94)
- Fix: Respect
332 Need account for login
response during FTP client authorization (thanks novak-as) - Fix: Set default value for
restartPosition
inDownload
method to matchFtpClient
method (thanks AdamLewisGMSL) - Fix: Improved handling for resuming upload when the connection drops (thanks pradu71)
- Fix: Honor custom parsers if server-specific handler is used
- Fix: Create server-specific handler if a custom handler has not been set (thanks Adhara3)
- New: Reworked timezone conversion API (simply set
TimeConversion
andTimeZone
) - New: Options to convert server timestamps into the format of your choice (
ServerTime
,LocalTime
andUTC
) - New: Support for conversion to local timezone in .NET core (set
LocalTimeZone
) - New:
GetListing
honors the time conversion settings of the active client - New:
GetModifiedDate
honors the time conversion settings of the active client - New:
SetModifiedDate
honors the time conversion settings of the active client - New: Reworked API for Custom file listing parsers (simply set
ListingCustomParser
on the client) - Fix: Drop support for legacy file listing parsing routines (
FtpParser.Legacy
will no longer work) - Fix: Unexpected time conversion occuring in
GetModifiedTimeAsync
- Change: Breaking changes to the
TimeConversion
property. - Change: Breaking changes to the
TimeOffset
property, which has been replaced byTimeZone
- Fix: "The connection was terminated before a greeting could be read"
- Platform: Add support for .NET Standard 2.1
- Platform: Upgrade
System.Net.Security
from version 4.3.0 to 4.3.2 to fix security issues - Performance: Quickly abort detection if host is unavailable during
AutoDetect
/AutoConnect
- Fix:
UploadDirectory
fails for some files with "Unable to read data from the transport connection" (thanks manuelxmarquez) - Fix: Comment for
UploadFile
/UploadFileAsync
/DownloadFile
/DownloadFileAsync
methods - Fix: Stack overflow during connection when server responds incorrectly to PASS command
- Fix: Uncatchable NullReferenceException is occasionally thrown from
ConnectAsync
- New: Automatic FTPS connection mode:
FtpEncryptionMode.Auto
which connects in FTP and attempts to upgrade to FTPS - New:
IsEncrypted
property to check if FTPS encryption is currently active for this connection - Fix:
ValidateCertificateRevocation
property was not being honored in async version (thanks kolorotur)
- Fix: Ensure file is retried sucessfully when first upload/download fails with an
IOException
(thanks manuelxmarquez) - Fix: Ensure file streams read and write correctly even when no
FtpClient
is provided (thanks manuelxmarquez) - Fix: Clear custom parser when removing parser or clearing all parsers (thanks rubenhuisman)
- New:
LocalFileBufferSize
property to control size of file buffer during local file I/O
- New:
UploadDirectoryDeleteExcluded
property to control if excluded files are deleted during Upload (thanks philippjenni) - New:
DownloadDirectoryDeleteExcluded
property to control if excluded files are deleted during Download (thanks philippjenni) - Fix: Dispose AsyncWaitHandles to stop handle leak in .NET Framework 4.5 (thanks sdiaman1)
- Fix: Implement proper cancellation support in
UploadDirectory
(once file transfer begins it cannot be cancelled) - Fix: Implement proper cancellation support in
DownloadDirectory
(once file transfer begins it cannot be cancelled) - Fix: Implement proper cancellation support in FXP
TransferDirectory
- Fix: Implement proper cancellation support in recursive
GetListing
- Fix: Correctly resume when unexpectedEOF error received during uploading a file (thanks mrcopperbeard)
- Fix: Hide internal properties in
FtpClient
that are not meant to be exposed - Fix: Update
IFtpClient
with the latest set of public properties that are meant to be exposed
- Fix: Downloading or uploading a directory can generate incorrect local paths
- Fix: Downloading or uploading a directory can generate incorrect local paths
- Fix: Expose
LoadProfile
API so it can be called by the generated code fromAutoDetect
- New:
ListingDataType
property to get file listings in ASCII/Binary - New:
DownloadZeroByteFiles
property to control if zero-byte files should be downloaded or skipped - Fix: Downloading 0-byte files crashes since no data downloaded
- New: All server-specific handlers moved to dedicated classes that extend
FtpBaseServer
- New: Ability to handle custom non-standard FTP servers by extending
FtpBaseServer
- Fix: Only overwrite local file after the first bytes downloaded of a remote file
- New: Tracking progress with FXP transfers is supported for all transfer modes
- New: Track low-level progress with new
TransferredBytes
inFtpProgress
class (thanks Adhara3) - New:
FXPProgressInterval
property to control how often FXP progress reports are sent - Fix: Hide
TransferFileFXPInternal
because its an internal transfer method and not to be used directly
- Fix:
FtpFileExtensionRule
was failing to compare extensions unless they were prefixed with a dot
- New:
GetChecksum
allows you to specify a hash algorithm to be run on the server if supported - New:
GetChecksum
has special support for switching the server-side algorithm for HASH command support - New: FXP file transfer now validates the file using the first mutually supported algorithm
- Fix: Incorrectly formatted string returned by utility method
TransferSpeedToString
- New:
CompareFile
andCompareFileAsync
methods to quickly perform various equality checks on a uploaded/downloaded file
- Fix: When download fails and we need to retry on failed verification, ensure that file is re-downloaded
- Fix: When FXP transfer fails and we need to retry on failed verification, ensure that file is re-transfered
- Fix: When uploaded file is skipped,
FtpStatus.Failed
is returned instead ofFtpStatus.Skipped
- Fix: Properly handle 4xx and 5xx series of errors and indicate failure when uploading or downloading files
- Fix: Correctly detect if server-side recursion is supported otherwise fallback to manual directory recursion
- Fix: Only resume download of files if Append mode is selected (in Overwrite mode we restart the download)
- Change:
Upload
andUploadAsync
now returnsFtpStatus
to indicate skipped, success or failed
- Fix: Proper session handling for FXP connections and disconnection of cloned connections
- Performance: Reduce redundant file size check in
DownloadFile
when appending is used
- New:
AutoDetect
andAutoConnect
now auto-configure for Azure FTP servers using known connection settings - Improve code generation of
FtpProfile
to use LoadProfile rather than setting each property individually - Add advanced Timeout and Socket settings to
FtpProfile
for Azure auto configuration - Fix: All exception classes now inherit from
FtpException
- All exceptions and
FtpProfile
are now serializable in .NET Framework
- New:
TransferFile
andTransferDirectory
methods to transfer files from server to server (thanks n0ix) - New: FXP (File Transfer Protocol) implementation to support direct server-to-server transfers (thanks n0ix)
- New: Predefined rules for filtering on file name using regular expressions (thanks n0ix)
- New: Predefined rules for filtering on folder name using regular expressions (thanks n0ix)
- Fix: Don't calculate ETA and percentage of
FtpProgress
if file size is zero (thanks Adhara3) - Fix:
GetFilePermissions
should useGetObjectInfo
instead ofGetListing
to prevent incorrect filepaths
- New: Support for MMD5 file hashing command to validate downloaded/uploaded files. (thanks n0ix)
- Change: Disable all
Begin*
andEnd*
methods for .NET 4.5 and onwards asasync
/await
is supported. - Improve:
GetHashAlgorithmAsync
andSetHashAlgorithmAsync
implemented as true async methods with cancellation support - Improve:
GetObjectInfoAsync
implemented as true async methods with cancellation support
- New: Download and upload file methods indicate if file was transferred, skipped or failed to transfer
- New: C# and VB.NET Examples for all file and folder transfer methods
- New: VB.NET Examples for all methods (not included in Nuget package but available on Github)
- Change:
DownloadFile
andUploadFile
returnFtpStatus
instead of boolean flag for tri-state feedback
- New: Support for XCRC FTP Command and CRC32 hash support to validate downloaded/uploaded files (thanks n0ix)
- Fix: Calculation of local file path during DownloadFolder sometimes ignores base directory
- New: Support multi-file progress tracking by indicating file index and local & remote path of the file
- New:
UploadDirectory
andDownloadDirectory
now supports tracking progress of the entire task - New:
UploadFiles
andDownloadFiles
now supports tracking progress for both sync/async methods - Fix: Update
IFtpClient
interface by adding newUploadDirectory
andDownloadDirectory
methods - Fix: Correctly determine file exists on servers that don't support SIZE command and return error 550
- Fix: Support more strings to determine if file exists using SIZE command
- New:
UploadDirectory
andUploadDirectoryAsync
methods to recursively upload or mirror a directory - New:
DownloadDirectory
andDownloadDirectoryAsync
methods to recursively download or mirror a directory - New: Rule engine to filter files that should be uploaded/downloaded according to multiple user-defined rules
- New: Predefined rules for filtering on folder name, useful for blacklisting certain system folders
- New: Predefined rules for filtering on file name or file extensions, useful for transferring a subset of files
- New: Predefined rules for filtering on file size, useful for filtering out very large files
- New: Ability to determine parent/self/child directories in listing using
SubType
property ofFtpListItem
- Fix: Machine listings sometimes cause infinite recursion in
GetListing
when recursing into self directory - Change:
CreateDirectory
andCreateDirectoryAsync
now return a flag indicating if it was created or skipped - Change: Use public fields instead of public properties for
FtpListItem
- Change: Improve performance of
CreateDirectory
by skipping the directory exists check
- Fix: Detect "file size not allowed in ASCII" string for French FTP servers
- Fix: TimeoutException when trying to read FTP server reply after Download/Upload
- New: Add
SendHost
andSendHostDomain
to control if HOST command is sent after handshake (thanks dansharpe83)
- Fix: Read stale NOOP responses after file transfer and also after
226 Transfer complete
(thanks aliquid) - Fix: Correct default value for
TimeConversion
property to assume UTC timestamps
- New: Support .NET Standard 2.0
- New: Keep control socket alive during long file transfers using NOOP (thanks aliquid)
- New: Add
NoopInterval
property to control interval of NOOP commands (thanks aliquid) - New: Add
TimeConversion
property to control if timestamps are converted from UTC into local time - Refactor: Rename
FtpExists
toFtpRemoteExists
to make its usage clear - New: Support detection of IBM z/OS and MVS FTP OS and server (allows for server specific commands)
- New: New constructors for
FtpClient
to support hostnames inUri
format - Fix: Always send progress reports after file download, even for zero-length files
- New:
ValidateCertificateRevocation
property to control if certificate revocation is checked.
- New:
ValidateAnyCertificate
property to validate any received server certificate, useful for Powershell - Fix: Default SSL protocol used in .NET 4.5+ release is now TLS 1.2 (latest supported protocol)
- New: Override the server-specific recursive LIST detection by setting
RecursiveList
- Fix typo in IP parsing regex that causes fallback to Host IP to fail (thanks Andy Whitfield)
- Fix: Verification of the MD5 Hash when file name contains spaces (thanks Nimelo)
- Fix: Safely absorb TimeoutException thrown after the file has fully uploaded/downloaded
- New: Progress reporting for synchronous methods
Upload
,Download
,UploadFile
andDownloadFile
are now sent via delegates - Fix: Correctly send progress for synchronous methods and retain
IProgress
for async methods
- Fix: Correctly assume Unix file listing parser for SunOS & Solaris servers
- Fix: Safely absorb TimeoutException thrown after the file has fully uploaded/downloaded
- New: Support detection of Sun OS Solaris FTP OS and server (allows for server specific commands)
- Fix: UploadFile fails when destination folder is empty on SunOS (550 error)
- Fix: Unable to upload files to OpenVMS servers if path contains numeric characters
- Fix: Assume FTP commands supported by OpenVMS HGFTP server if FEAT not supported
- FiX: Improve detection of OpenVMS absolute paths
- Fix:
Connect
&ConnectAsync
throw ArgumentException when passing an incompleteFtpProfile
- New: Auto-detect the correct FTP listing parser when SYST command fails (IIS, Azure, OpenVMS)
- New: Assume FTP commands supported by OpenVMS HGFTP server
- FiX: Support edge case for OpenVMS absolute paths (directive can be alpha-numeric)
- New: Improved transfer rate throttling when using an upload/download speed limit (thanks wakabayashik)
- New: Support detection of XLight FTP server software (allows for server specific commands)
- New: Partial support for getting directory listing using STAT command (
GetListing
supports newFtpListOption.UseStat
) - Fix:
GetFileSize
always returns 0 instead of correct file size (thanks RadiatorTwo)
- Fix:
FileExists
andFileExistsAsync
support switching to binary mode for servers that need it
- Fix: Error using BlueCoat proxy to an FTP server on a port other than port 21
- Fix: Error using UserAtHost proxy to an FTP server on a port other than port 21
- New: Change
Capability
API to return a list instead of bitwise enum (to support more than 32 distinct capabilities) - New: Change custom parsers to take capabilities as a list instead of bitwise enum (to match client implementation)
- New: Support detection of FTP2S3 gateway server software (allows for server specific commands)
- New: Support detection of server-specific capabilities of Serv-U FTP Gateway
- New: Support
RMDA
command to quickly and recursively delete a directory from Serv-U FTP Gateway
- Fix: Improve performance of
GetFileSize
to only switch to Binary for servers that require it - Fix: Ensure data type (ASCII/Binary) is correctly set during
GetFileSize
for servers that require it - Fix: Ensure data type (ASCII/Binary) is correctly set for cloned connections
- Fix: Ensure data type (ASCII/Binary) is correctly set during
GetListing
andGetNameListing
- Fix: Reset server detection state flags whenever we connect to a server, to allow for reuse of
FtpClient
- Fix: Copy server detection state flags to cloned connections to improve performance
- Fix: Retry
GetListing
if temporary error "Received an unexpected EOF or 0 bytes from the transport stream"
- Fix: Prefer using Passive/Active modes rather than Enhanced Active/Passive during auto-detection
- Fix: Some FTP servers do not open a port when listing an empty folder with
GetNameListing
- Fix: Hard catch and suppress all exceptions during disposing to solve all random exceptions
- New: Automatic FTP connection negotiation using
AutoConnect()
- New: Automatic detection of working FTP connection settings using
AutoDetect()
- New: C# code generation of working connection settings using
FtpProfile.ToCode()
- New: Support more capability detection commands: EPSV, CPSV, NOOP, CLNT, SSCN, SITE commands for ProFTPd
- New: Improve transfer performance by only attempting EPSV once and then never using it again for that connection
- New: Support MKDIR & RMDIR commands specially for ProFTPd to quickly create and delete a directory on the server-side
- New: Support PRET command before downloading or uploading files for servers like ProFTPd & DrFTPd
- New: Support detection of BFTPd server software (allows for server specific commands)
- Fix: When uploading files in
FtpExists.NoCheck
mode, file size check should not be done - Fix: Some FTP servers do not open a port when listing an empty folder (thanks Mortens4444)
- Fix:
OpenRead
withEnableThreadSafeDataConnections
always transfers in ASCII (thanks ts678) - Refactor: Delete legacy static methods:
OpenRead
,OpenWrite
,OpenAppend
(dynamic versions still exist) - Refactor: Move
CalcChmod
fromFtpClient
toFtpExtensions
(as part of repository cleanup task)
- Fix: Async methods do not work with Active FTP mode and SSL/encryption (thanks Mortens4444)
- Fix: For OpenVMS absolute paths may not contain slashes but are still absolute (3rd revision)
- Fix: Divide-by-zero exceptions while calculating progress of file uploads/downloads
- Fix: Supress all exceptions when Disposing the underlying FtpSocketStream
- Fix: Received an unexpected EOF or 0 bytes from the transport stream (thanks mikemeinz)
- Fix:
UploadFile()
progress callback is not called if the file already exists on the server - (.NET core) Fix:
Connect()
method sometimes causes the thread to hang indefinitely (thanks radiy) - Fix: Regression of #288 where upload hangs with only a few bytes left (thanks cw-andrews)
- New:
FtpAuthenticationException
for authentication errors (thanks erik-wramner) - New: Added support to detect Homegate FTP Server
- New: SSL Buffering is now switchable via the
SslBuffering
parameter - Fix: SSL Buffering is automatically disabled when using FTP proxies, and enabled in all other cases
- Fix: Revert PR #383 as it was causing regression issues in SSL connectivity
- Fix: Disable automatic IP correction to fix connectivity issues via BlueCoat proxy servers (thanks CMIGIT)
- Refactor: Rename
FtpClientUserAtHostProxyBlueCoat
toFtpClientBlueCoatProxy
- Fix: For OpenVMS absolute paths may not contain slashes but are still absolute (2nd revision) (thanks tonyhawe)
- Fix: Detect file existence string
"Can't find file"
to fix FileExists check on some servers (thanks reureu) - Fix: Feature parity between
FileExists
andFileExistsAsync
methods, added support for FtpReply 550 check (thanks reureu) - Fix: Feature parity between
UploadFile
andUploadFileAsync
methods, added support for AppendNoCheck handling (thanks reureu)
- New: Get detailed progress information for uploads/downloads via the
FtpProgress
object (thanks n0ix) - New: Get transfer speed and ETA (estimated time of arrival) for uploads/downloads (thanks n0ix)
- Fix: Files were uploaded in Write mode instead of Append mode when the exists mode is
AppendNoCheck
and we couldn't read the offset position (thanks @everbalovas) - Fix: Swap
SslStream
andBufferedStream
so proxied connections withFtpClientHttp11Proxy
are to connect (thanks @rmja)
- New: Additional FTP Server software detection (HP NonStop/Tandem, GlobalScape EFT, Serv-U, Cerberus, CrushFTP, glFTPd)
- New: Assume capabilities for servers that don't support FEAT (wuFTPd)
- Fix:
FileExists
returns false if name listing is used and server lists filenames with the path - Fix: For OpenVMS absolute paths may not contain slashes but are still absolute
- Fix: For
Download()
methodsrestartPosition
should not be mandatory
- New: Ability to cancel all async methods via
CancellationToken
(thanks WolfspiritM) - New:
ReadTimeout
is now honored by all async methods (thanks WolfspiritM) - New: FTP Server operating system detection (Windows, Unix, VMS, IBM/OS400)
- (.NET core) Fix: GetListing blocking with no timeout (thanks WolfspiritM)
- (.NET core) Fix async methods by not using the the async read function (thanks WolfspiritM)
- New: Ability to resume a download via
existsMode
onDownloadFile()
andDownloadFiles()
(thanks n0ix) - New: Ability to turn off checking for server capabilities using FEAT command (thanks nhh-softwarehuset)
- Fix: Add workaround if a server advertises a non-routeable IP in PASV Mode (thanks n0ix)
- Fix: Recursive directory deletion tries to delete the same file twice (because GetListing is also recursive)
- New:
OnLogEvent
callback to get logs in the context of indivivdual FtpClient connections - Fix: All logging is done in the context of an
FtpClient
and then passed toFtpTrace
listeners - Signature for custom list parsers has changed,
FtpClient
argument added to the end
- New: FTP Server software detection (PureFTPd, VsFTPd, ProFTPD, FileZilla, OpenVMS, WindowsCE, WuFTPd)
- New: Detect if the FTP server supports recursive file listing (LIST -R) command using whitelist
- New:
GetListing
will manually recurse through directories ifFtpListOption.Recursive
is set and server does not support recursion - New: Added
LastReply
property which returns the lastFtpReply
recieved from the server. - New: Added new upload option
AppendNoCheck
to append to a file on the server without checking if it exists (thanks @everbalovas) - Fix: During upload, respond to any error in 5xx series, not just 550 (thanks stengnath)
- Fix: Various fixes to
UploadFileAsync
based on fixes already implemented inUploadFile
- Fix:
UploadFilesAsync
witherrorHandling
deletes the entire directory instead of specific files - Fix: Server responds to EPSV with 425 "Data connection failed" but connects with PASV (thanks ejohnsonTKTNET)
- Fix: Use proper async configuration for .NET Async methods (thanks ejohnsonTKTNET)
- Fix: Improve implementation of upload and download resuming in Async methods (thanks ejohnsonTKTNET)
- Fix:
UploadFile()
orUploadFiles()
sometimes fails to create the remote directory if it doesn't exist - Fix:
DownloadDataType
Binary value ignored on ASCII-configured FTP servers - Performance improvement: Added
BufferedStream
betweenSslStream
andNetworkStream
(thanks stengnath) - Fix: When the FTP server sends 550, transfer is received but not confirmed (thanks stengnath)
- Fix: Make
Dispose
method ofFTPClient
virtual (thanks @martinbu) - Fix:
OpenPassiveDataStream
/Async()
uses the target FTP host instead of the configured proxy (thanks @rmja) - Fix:
FileExists()
for Xlight FTP Server (thanks @oldpepper) - Fix: FTPD "550 No files found" when folder exists but is empty, only in PASV mode (thanks stengnath)
- Fix: Many unexpected EOF for remote file
IOException
on Android (thanks @jersiovic) - Fix: Race condition when
BeginInvoke
calls the callback before theIAsyncResult
is added (thanks stengnath)
- Fix: Prevent socket poll from hammering the server multiple times per second
- Fix: Allow using absolute paths that include drive letters (Windows servers)
- Performance improvement: Only change the FTP data type if different from required type
- Performance improvement: Download all files in EOF mode and skip the file size check, unless download progress is required
- Added all missing async versions of FTP methods to
IFtpClient
- System: Certain core FTP socket handling operations have been changed to improve reliability & performance.
- Fix: Fix hang in TLS activation because no timeout is set on the underlying
NetworkStream
(thanks @iamjay)
- Added async versions of FTP methods to
IFtpClient
(thanks @peterfortuin) - Fix: Fixes when
ActivePorts
is specified in active FTP mode (thanks @ToniMontana) - Fix: Throw
OperationCanceledException
instead ofFtpException
when cancellation is requested (thanks taoyouh)
- Fix: Add support for checking if file exists on Serv-U FTP Server
- Fix: Make
IFtpClient
inherit fromIDisposable
(thanks @repl-andrew-ovens) - (UWP) Fix: UWP does not allow
File.Exists()
to run in UI thread (thanks taoyouh)
- Fix: When downloading files in ASCII mode, file length is unreliable therefore we read until EOF
- Fix: When upload/download progress is indeterminate, send -1 instead of NaN or Infinity
- Fix:
NetStream
was not assigned inFtpSocketStream
for .NET Standard in active FTP mode (thanks @ralftar) - Fix:
CurrentDataType
was not set for ASCII transfers inDownloadFileAsync
/UploadFileAsync
(thanks taoyouh) - Fix: Sometimes
FtpSocketStream
andFtpDataStream
are not disposed inFtpSocketStream.Dispose
(thanks taoyouh)
- New Progress reporting for
UploadFile
&DownloadFile
methods viaIProgress
- Fix:
Stream.Position
should not be set inUploadFileInternal
unless supported
- New Task-based async methods for .NET Standard and .NET Fx 4.5 (thanks taoyouh)
- New async methods for
UploadFile
,DownloadFile
,UploadFiles
&DownloadFiles
(thanks artiomchi) - (UWP) Fix:
FileNotFoundException
with referenceSystem.Console
(thanks artiomchi) - (.NET core) Fix: Thread suspends when calling
UploadFile
orDownloadFile
(thanks artiomchi) - (.NET core) Fix: File download hangs inconsistently when reading data from stream (thanks @artiomchi, bgroenks96)
- (.NET core) Fix: Stream does not dispose due to wrong handling of closing/disposing (thanks artiomchi)
- Fix: File upload EOS bug when calling
Stream.Read
(thanks bgroenks96, @artiomchi, @taoyouh) - Fix:
DownloadFileInternal
not recognizing the download data type withEnableThreadSafeConnections
(thanks bgroenks96) - (Backend) Migrate to a single VS 2017 solution for all frameworks (thanks artiomchi)
- (Backend) Continuous Integration using AppVeyor (thanks artiomchi)
- Add
IFtpClient
interface to build unit tests upon mainFtpClient
class (thanks Kris0) - Disposing
FtpDataStream
reads server reply and closes the underlying stream (thanks Lukazoid)
- New
SetModifiedTime
API to change modified date of a server file in local timezone/UTC - Add type argument to
GetModifiedTime
, allowing for getting dates in UTC/Local timezone - Breaking changes to Async API of
GetModifiedTime
(addition of type argument) GetModifiedTime
andSetModifiedTime
now honor theTimeOffset
property inFtpClient
- Add
checkIfFileExists
toOpenRead
,OpenAppend
andOpenWrite
to skipGetFileSize
check - Fix issue where
InnerException
is null during a file transfer (upload/download) - Improve performance of typical uploads/downloads by skipping the extra file exists check
- Fix for
CreateDirectory
andDirectoryExists
to allow null/blank input path values - Fix for
GetFtpDirectoryName
to return correct parent folder of simple folder paths (thanks ww898)
- Add argument validation for missing/blank arguments in :
Upload, Download, UploadFile(s), DownloadFile(s), GetObjectInfo, DeleteFile, DeleteDirectory, FileExists, DirectoryExists, CreateDirectory, Rename, MoveFile, MoveDirectory, SetFilePermissions, Chmod, GetFilePermissions, GetChmod, GetFileSize, GetModifiedTime, VerifyTransfer, OpenRead, OpenWrite, OpenAppend
- Disable all async methods on .NET core due to persistant
PlatformUnsupported
exception (if you need async you are free to contribute a non-blocking version of the methods)
- Increase performance of
GetListing
by reading multiple lines at once (BulkListing property, thanks sierrodc)
- Add support for parsing AS400 listings inside a file (5 fields) (thanks rharrisxtheta)
- Retry interpreting file listings after encountered invalid date format (thanks rharrisxtheta)
- Always switch into binary mode when running SIZE command (thanks rharrisxtheta)
- Honor
UploadDataType
andDownloadDataType
in all sync/async cases (thanks rharrisxtheta) - Force file transfers in BINARY mode for known 0 byte files (thanks rharrisxtheta)
- Allow file transfers in ASCII mode if the server doesn't support the SIZE command (thanks rharrisxtheta)
- Fix
NullReferenceException
when arguments are null duringFtpTrace.WriteFunc
- Remove internal locking for .NET Standard 1.4 version since unsupported on UWP
- Remove dependency on
System.Threading.Thread
for .NET Standard 1.4 version (for UWP)
- Allow transferring files in ASCII/Binary mode with the high-level API (UploadDataType, DownloadDataType)
- Add support for .NET 3.5 and .NET Standard 1.4 (supports Universal Windows Platform 10.0)
- Add
FtpTrace.LogToConsole
andLogToFile
to control logging in .NET core version
- Add
PlainTextEncryption
API to support FTPS servers and plain-text FTP firewalls (CCC command) - FluentFTP now uses unsafe code to support the CCC command (inside
FtpSslStream
) - If you need a "non unsafe" version of the library please add an issue
- Add logging for high-level function calls to improve remote debugging (
FtpTrace.LogFunctions
) - Add settings to hide sensitive data from logs (
FtpTrace.LogIP
,LogUserName
,LogPassword
) - Add
RecursiveList
to control if recursive listing should be used - Auto-detect Windows CE and disable recursive listing during
DeleteDirectory()
- Add
UploadRateLimit
andDownloadRateLimit
to control the speed of data transfer (thanks Danie-Brink)
- Fix parsing of
LinkTarget
duringGetListing()
on Unix FTP servers - Improve logging clarity by removing "FluentFTP" prefix in TraceSource
- Add
MoveFile()
andMoveDirectory()
to move files and directories safely
- Automatically verify checksum of a file after upload/download (thanks jblacker)
- Configurable error handling (abort/throw/ignore) for file transfers (thanks jblacker)
- Multiple log levels for tracing/logging debug output in
FtpTrace
(thanks jblacker)
- Simplify
DeleteDirectory()
API - theforce
andfastMode
args are no longer required DeleteDirectory()
is faster since it uses one recursive file listing instead of many- Remove .NET Standard 1.4 to improve nuget update reliability, since we need 1.6 anyway
- Split stream API into
Upload()
/UploadFile()
andDownload()
/DownloadFile()
- Greatly improve performance of
FileExists()
andGetNameListing()
- Add new OS-specific directory listing parsers to
GetListing()
andGetObjectInfo()
- Support
GetObjectInfo()
even if machine listings are not supported by the server - Add
existsMode
toUploadFile()
andUploadFiles()
allowing for skip/overwrite and append - Remove all usages of string.Format to fix reliability issues caused with UTF filenames
- Fix issue of broken files when uploading/downloading through a proxy (thanks Zoltan666)
GetReply()
is now public so users ofOpenRead
/OpenAppend
/OpenWrite
can call it after
- Add async/await support to all methods for .NET 4.5 and onwards (thanks jblacker)
- Support for .NET Standard 1.4 added.
- Add
UploadFiles()
andDownloadFiles()
which is faster than single file transfers - Allow disabling UTF mode using DisableUTF8 API
- First .NET Core release (DNXCore5.0) using Visual Studio 2017 project and shared codebase.
- Support for .NET 2.0 also added with shims for LINQ commands needed.
- Add
FtpListOption.IncludeSelfAndParent
toGetListing
- Use streams during upload/download of files to improve performance with large files
- Support for uploading/downloading to Streams and byte[] with
UploadFile()
andDownloadFile()
- Added high-level
UploadFile()
andDownloadFile()
API. Fixed some race conditions.
- Added support for FTP proxies using HTTP 1.1 and User@Host modes. (thanks L3Z4)