使用WinSCP从SFTP服务器下载超过X天的文件,跳过不包含任何匹配文件的文件夹

时间:2017-08-30 16:21:17

标签: sftp winscp

我仅限于PuTTY和WinSCP。

我正在尝试使用日志文件下载日志目录。例如,我想抓住所有log_files 6天或更早的人。包含文件夹的log_dir2log_dir3符合条件,而log_dir1及其文件则不符。

DIR/log_dir1/log_files % older than 6 days
DIR/log_dir2/log_files % meets criteria
DIR/log_dir3/log_files % meets criteria

我的问题是,虽然未下载log_files log_dir1,但我目前使用的语法会下载log_dir1文件夹。通常,这不是什么大问题,但我们正在谈论数百个log_dir个文件夹(由于文件超过6天,所有文件夹都是空的)。由于我无法控制的原因,我无法使用其日志文件移动或归档这些旧日志目录。

我的问题很简单,如何更改语法以忽略超过6天的文件夹以及文件。

get -filemask="*>6D" /DIR/* C:\temp

我已经尝试了几种不同的参数组合,我已阅读有关目录掩码和路径掩码的支持页面。我不能让他们中的任何一个工作(版本问题?)。任何人都可以比help page更好地解释他们的语法。我将在明天使用我正在使用的当前版本的WinSCP进行更新。

1 个答案:

答案 0 :(得分:0)

Time constraint in WinSCP file mask不能用于目录。

但您可以在PowerShell script with a use of WinSCP .NET assembly中轻松实现此自定义逻辑:

# Load WinSCP .NET assembly
Add-Type -Path "WinSCPnet.dll"

# Set up session options
$sessionOptions = New-Object WinSCP.SessionOptions -Property @{
    Protocol = [WinSCP.Protocol]::Sftp
    HostName = "example.com"
    UserName = "username"
    Password = "password"
    SshHostKeyFingerprint = "..."
}

$remotePath = "/remote/path"
$localPath = "C:\local\path"
$limit = (Get-Date).AddDays(-6)

$session = New-Object WinSCP.Session

# Connect
$session.Open($sessionOptions)

# Enumerate files to download
$fileInfos =
    $session.EnumerateRemoteFiles(
        $remotePath, $Null, [WinSCP.EnumerationOptions]::AllDirectories) |
    Where-Object { $_.LastWriteTime -gt $limit }

foreach ($fileInfo in $fileInfos)
{
    $localFilePath =
        [WinSCP.RemotePath]::TranslateRemotePathToLocal(
            $fileInfo.FullName, $remotePath, $localPath)

    # If the corresponding local folder does not exist yet, create it
    $localFileDir = Split-Path -Parent $localFilePath
    if (!(Test-Path -Path $localFileDir))
    {
        Write-Host "Creating local directory $localFileDir..."
        New-Item $localFileDir -ItemType directory | Out-Null
    }

    Write-Host "Downloading file $($fileInfo.FullName)..."

    # Download file
    $sourcePath = [WinSCP.RemotePath]::EscapeFileMask($fileInfo.FullName)
    $transferResult = $session.GetFiles($sourcePath, $localFilePath)

    # Did the download succeeded?
    if (!$transferResult.IsSuccess)
    {
        # Print error (but continue with other files)
        Write-Host ("Error downloading file ${remoteFilePath}: " +
            $transferResult.Failures[0].Message)
    }
}

$session.Dispose()

Write-Host "Done."

运行脚本(download.ps1),如:

powershell.exe -ExecutionPolicy Unrestricted -File download.ps1