【问题标题】:PowerShell How can I delete a directory with FTP?PowerShell 如何使用 FTP 删除目录?
【发布时间】:2018-10-13 07:02:34
【问题描述】:

从 Web 请求方法字段的 this list 中,我已经通过 FTP 成功地从我的服务器中删除了文件,基本上是在 this page 上填写指南并稍作改动:

$url = "ftp://server.net/path/to/place/FILE.txt"
$userName = "username"
$password = "p@ssw0rd!"

$ftpreq = [System.Net.FtpWebRequest]::create($url)
$ftpreq.Credentials = New-Object System.Net.NetworkCredential($userName, $password)
$ftpreq.Method = [System.Net.WebRequestMethods+Ftp]::DeleteFile

$ftpreq.GetResponse()

文件名为FILE.txt,文件夹名为FOLDER

当我尝试对文件夹执行类似操作时,我收到 PowerShell 错误和 550 响应。

我基本上尝试了两种文件删除方法。

尝试 1:更改 URL 以匹配文件夹名称,保留方法

请注意,我尝试删除的文件夹位于我已成功删除的文件旁边。

$url = "ftp://server.net/path/to/place/FOLDER"
$userName = "username"
$password = "p@ssw0rd!"

$ftpreq = [System.Net.FtpWebRequest]::create($url)
$ftpreq.Credentials = New-Object System.Net.NetworkCredential($userName, $password)
$ftpreq.Method = [System.Net.WebRequestMethods+Ftp]::DeleteFile

$ftpreq.GetResponse()

使用“0”参数调用“GetResponse”的异常:“远程服务器返回错误:(550) 文件不可用 (例如,找不到文件,无法访问)。

尝试2:改变方法

$url = "ftp://server.net/path/to/place/FOLDER"
$userName = "username"
$password = "p@ssw0rd!"

$ftpreq = [System.Net.FtpWebRequest]::create($url)
$ftpreq.Credentials = New-Object System.Net.NetworkCredential($userName, $password)
$ftpreq.Method = [System.Net.WebRequestMethods+Ftp]::RemoveDirectory

$ftpreq.GetResponse()

使用“0”参数调用“GetResponse”的异常:“远程服务器返回错误:(550) 文件不可用 (例如,找不到文件,无法访问)。”


已实施的解决方案

function DeleteFtpFolder($url, $credentials)
{
    $listRequest = [Net.WebRequest]::Create($url)
    $listRequest.Method = [System.Net.WebRequestMethods+FTP]::ListDirectoryDetails
    $listRequest.Credentials = $credentials

    $lines = New-Object System.Collections.ArrayList

    $listResponse = $listRequest.GetResponse()
    $listStream = $listResponse.GetResponseStream()
    $listReader = New-Object System.IO.StreamReader($listStream)

    while (!$listReader.EndOfStream)
    {
        $line = $listReader.ReadLine()
        $lines.Add($line) | Out-Null
    }

    $listReader.Dispose()
    $listStream.Dispose()
    $listResponse.Dispose()

    foreach ($line in $lines)
    {
        $tokens = $line.Split(" ", 5, [System.StringSplitOptions]::RemoveEmptyEntries)

        $type = $tokens[2]
        $name = $tokens[3]
        $fileUrl = ($url + "/" + $name)

        if ($type -eq "<DIR>")
        {
            Write-Host "Found folder: $name"

            DeleteFtpFolder $fileUrl $credentials

            Write-Host "Deleting folder: $name"
            $deleteRequest = [Net.WebRequest]::Create($fileUrl)
            $deleteRequest.Credentials = $credentials
            $deleteRequest.Method = [System.Net.WebRequestMethods+FTP]::RemoveDirectory
            $deleteRequest.GetResponse() | Out-Null
        }
        else 
        {
            $fileUrl = ($url + "/" + $name)
            Write-Host "Deleting file: $name"

            $deleteRequest = [Net.WebRequest]::Create($fileUrl)
            $deleteRequest.Credentials = $credentials
            $deleteRequest.Method = [System.Net.WebRequestMethods+FTP]::DeleteFile
            $deleteRequest.GetResponse() | Out-Null
        }
    }
}

$credentials = New-Object System.Net.NetworkCredential($AzureFtpUsername, $AzureFtpPassword)
$url = $AzureFtpUrl

DeleteFtpFolder $url $credentials

设置是嵌入在 Octopus Deploy 内的流程步骤中的 PowerShell 脚本。这就是我调用底部函数的原因。

此解决方案与accepted answer 几乎相同,只是对我放置递归调用的位置以及如何解析从服务器返回的数据进行了一些小的改动。他的解决方案看起来更像ls 输出,列更多,而我的解决方案看起来更像dir 输出,列更少。但即便如此,它看起来也不像我本地 Windows 机器上的 dir 输出,所以我不确定到底发生了什么。但它有效,所以这就足够了。

【问题讨论】:

    标签: .net powershell ftp ftpwebrequest


    【解决方案1】:

    RMD FTP 命令(RemoveDirectory 方法)失败,如果目录不为空。

    通常它会失败并出现如下错误:

    550 目录不为空。

    不幸的是,FtpWebRequest 有一个将 FTP 错误代码“翻译”为自己的消息的坏习惯。在这种情况下,它将 550 “翻译”为:

    文件不可用(例如,文件未找到、无法访问)。

    什么隐藏了真正的问题。


    无论如何,FtpWebRequest class(或 .NET 框架中的任何其他 FTP 实现)中不支持递归操作。您必须自己实现递归:

    • 列出远程目录
    • 迭代条目、删除文件并递归到子目录(再次列出它们等)

    棘手的部分是从子目录中识别文件。使用FtpWebRequest 无法以便携的方式做到这一点。 FtpWebRequest 不幸的是不支持 MLSD 命令,这是在 FTP 协议中检索具有文件属性的目录列表的唯一可移植方式。另见Checking if object on FTP server is file or directory

    您的选择是:

    • 对文件名执行操作,该操作对于文件肯定会失败,而对于目录肯定会成功(反之亦然)。 IE。您可以尝试下载“名称”。如果成功,它是一个文件,如果失败,它是一个目录。但是,当您有大量条目时,这可能会成为性能问题。
    • 您可能很幸运,在您的特定情况下,您可以通过文件名来区分目录中的文件(即,您的所有文件都有扩展名,而子目录没有)
    • 您使用长目录列表(LIST 命令 = ListDirectoryDetails 方法)并尝试解析特定于服务器的列表。许多 FTP 服务器使用 *nix 样式的列表,您可以在其中通过条目开头的 d 标识目录。但是许多服务器使用不同的格式。以下示例使用这种方法(假设为 *nix 格式)。
    • 在这种特定情况下,您可以尝试将条目作为文件删除。如果删除失败,请尝试将条目列为目录。如果列表成功,您假设它是一个文件夹并相应地继续。不幸的是,当您尝试列出文件时,某些服务器不会出错。他们将只返回一个包含该文件的单个条目的列表。
    function DeleteFtpFolder($url, $credentials)
    {
        $listRequest = [Net.WebRequest]::Create($url)
        $listRequest.Method = [System.Net.WebRequestMethods+Ftp]::ListDirectoryDetails
        $listRequest.Credentials = $credentials
    
        $lines = New-Object System.Collections.ArrayList
    
        $listResponse = $listRequest.GetResponse()
        $listStream = $listResponse.GetResponseStream()
        $listReader = New-Object System.IO.StreamReader($listStream)
        while (!$listReader.EndOfStream)
        {
            $line = $listReader.ReadLine()
            $lines.Add($line) | Out-Null
        }
        $listReader.Dispose()
        $listStream.Dispose()
        $listResponse.Dispose()
    
        foreach ($line in $lines)
        {
            $tokens = $line.Split(" ", 9, [StringSplitOptions]::RemoveEmptyEntries)
            $name = $tokens[8]
            $permissions = $tokens[0]
    
            $fileUrl = ($url + $name)
    
            if ($permissions[0] -eq 'd')
            {
                DeleteFtpFolder ($fileUrl + "/") $credentials
            }
            else
            {
                Write-Host "Deleting file $name"
                $deleteRequest = [Net.WebRequest]::Create($fileUrl)
                $deleteRequest.Credentials = $credentials
                $deleteRequest.Method = [System.Net.WebRequestMethods+Ftp]::DeleteFile
                $deleteRequest.GetResponse() | Out-Null
            }
        }
    
        Write-Host "Deleting folder"
        $deleteRequest = [Net.WebRequest]::Create($url)
        $deleteRequest.Credentials = $credentials
        $deleteRequest.Method = [System.Net.WebRequestMethods+Ftp]::RemoveDirectory
        $deleteRequest.GetResponse() | Out-Null
    }
    

    使用如下函数:

    $url = "ftp://ftp.example.com/path/to/folder/";
    $credentials = New-Object System.Net.NetworkCredential("username", "password")
    DeleteFtpFolder $url $credentials
    

    或者使用支持递归操作的 3rd 方库。

    例如使用WinSCP .NET assembly,您只需调用Session.RemoveFiles即可删除整个目录:

    # Load WinSCP .NET assembly
    Add-Type -Path "WinSCPnet.dll"
    
    # Setup session options
    $sessionOptions = New-Object WinSCP.SessionOptions -Property @{
        Protocol = [WinSCP.Protocol]::Ftp
        HostName = "ftp.example.com"
        UserName = "username"
        Password = "password"
    }
    
    $session = New-Object WinSCP.Session
    
    # Connect
    $session.Open($sessionOptions)
    
    # Remove folder
    $session.RemoveFiles("/path/to/folder").Check()
    
    # Disconnect, clean up
    $session.Dispose()
    

    如果服务器支持,WinSCP 在内部使用MLSD 命令。如果没有,它使用LIST 命令并支持数十种不同的列表格式。

    (我是 WinSCP 的作者)

    【讨论】:

    • 非常感谢。我能够使用您发布的递归解决方案进行一些小的改动来解决问题。我从服务器返回的输出看起来更像dir 而不是ls,所以我编写了代码来查找字符串&lt;DIR&gt; 来区分文件和文件夹。
    • 解析 DOS 列表(可能是 IIS 服务器)的代码可以在:stackoverflow.com/q/7060983/850848#39771146 找到(虽然它在 C# 中,但转换为 PowerShell 应该不难)。尽管您还可以将 IIS 配置为使用 *nix/ls 样式列表。
    猜你喜欢
    • 2017-07-05
    • 2012-05-31
    • 1970-01-01
    • 2010-12-28
    • 2010-12-17
    • 1970-01-01
    • 2021-02-19
    • 1970-01-01
    相关资源
    最近更新 更多