r12*_*r12 5 .net ftp powershell ftpwebrequest
从这个 Web请求方法字段列表中,我已经成功地通过FTP从我的服务器中删除了文件,基本上填写了本页面上的指南,并做了一些小改动:
$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()
Run Code Online (Sandbox Code Playgroud)
该文件已命名FILE.txt
,该文件夹已命名FOLDER
.
当我尝试做类似的事情,但对于一个文件夹,我得到PowerShell错误和550响应.
我基本上尝试了两种文件删除方法.
请注意,我尝试删除的文件夹位于我已成功删除的文件旁边.
$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()
Run Code Online (Sandbox Code Playgroud)
使用"0"参数调用"GetResponse"的异常:"远程服务器返回错误:(550)文件不可用(例如,找不到文件,没有访问权限).
$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()
Run Code Online (Sandbox Code Playgroud)
使用"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
Run Code Online (Sandbox Code Playgroud)
该设置是Octopus Deploy内部流程步骤中的嵌入式PowerShell脚本.这就是我在底部调用函数的原因.
此解决方案几乎与接受的答案相同,对递归调用的位置以及如何解析来自服务器的返回数据进行了一些小的改动.他的解决方案看起来更像是一个ls
输出,有更多的列,而我看起来更像是一个dir
输出,列数更少.但即便如此,它看起来并不像dir
我本地Windows机器上的输出,所以我不确定究竟发生了什么.但它确实有效,所以这已经足够了.
RMD
RemoveDirectory
如果目录不为空,则FTP命令(方法)失败.
通常它失败并出现如下错误:
550目录不为空.
不幸的是,FtpWebRequest
有一个坏习惯是将FTP错误代码"翻译"到自己的消息中.在这种情况下,它"翻译"550到:
文件不可用(例如,找不到文件,没有访问权限).
什么隐藏了真正的问题.
无论如何,有在递归操作不支持FtpWebRequest
类(或在.NET框架的任何其他FTP实现).你必须自己实现递归:
棘手的部分是识别子目录中的文件.用便携式方式无法做到这一点FtpWebRequest
.该FtpWebRequest
遗憾的是不支持的MLSD
命令,这是检索目录列表中的FTP协议与文件属性的唯一可移植的方法.另请参阅检查FTP服务器上的对象是文件还是目录.
你的选择是:
LIST
command = ListDirectoryDetails
method)并尝试解析特定于服务器的列表.许多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
}
Run Code Online (Sandbox Code Playgroud)
使用如下功能:
$url = "ftp://ftp.example.com/path/to/folder/";
$credentials = New-Object System.Net.NetworkCredential("username", "password")
DeleteFtpFolder $url $credentials
Run Code Online (Sandbox Code Playgroud)
或者使用支持递归操作的第三方库.
例如,使用WinSCP .NET程序集,只需调用一次即可删除整个目录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()
Run Code Online (Sandbox Code Playgroud)
在内部,MLSD
如果服务器支持,WinSCP将使用该命令.如果没有,它使用该LIST
命令并支持许多不同的列表格式.
(我是WinSCP的作者)
归档时间: |
|
查看次数: |
953 次 |
最近记录: |