Как получить сводную информацию о Robocopy, которая входит в PowerShell?

У меня есть быстрый скрипт, написанный для сравнения различий в файлах на разных серверах, а затем я получаю подсказку о том, следует ли отражать изменения Я бы хотел, чтобы он не запрашивал копирование, если в нем нет изменений.

Сценарий:

robocopy "$cSource" "$cDestination" /E /L /FP

$cDecision = Read-host "Continue = 1, anything else = Stop"

if ($cDecision -eq "1") {
robocopy "$cSource" "$cDestination" /MIR /FFT /Z /W:5 /MT:64 /XX /log:$cLogLocation

Invoke-item $cLogLocation

}

else { Write-host "Quit!" }

Я бы хотел, чтобы Powershell не просил продолжить, если Robocopy сообщит 0 для Несоответствия, Неудачного, Дополнительного и Скопированного. Это не очень важно для этого конкретного проекта, но я могу вспомнить пару полезных применений для такой проверки. Спасибо!

3 ответа

Решение

Наконец-то люди не ответили за 10 секунд! Я использовал вашу проблему, чтобы продолжить узнавать больше о Powershell. Следующее работает для меня, я надеюсь увидеть, как другие уменьшат этот код:

Clear-Host
$ErrorActionPreference = "Continue"
$DebugPreference = "Continue"
$VerbosePreference = "Continue"

@"

## robocopy_helper.ps1 ########################################################
Usage:        powershell -ExecutionPolicy Bypass -File ./robocopy_helper.ps1

Purpose:      Dry run before Full run of robocopy

History:      07/11/2014  -  Created
###############################################################################

"@

## User Supplied Variables
$cSrc = "C:\Temp"
$cDst = "C:\Temp2"
$cLog = "c:\robo.log"

## Robocopy Dry Run
$robo_test = robocopy "$cSrc" "$cDst" /E /L /FP

## Use Regular Expression to grab the following Table
#               Total    Copied   Skipped  Mismatch    FAILED    Extras
#    Dirs :         1         0         1         0         0         0
#   Files :         1         0         1         0         0         0
$robo_results = $robo_test -match '^(?= *?\b(Total|Dirs|Files)\b)((?!    Files).)*$'

## Convert Table above into an array
$robo_arr = @()
foreach ($line in $robo_results){
    $robo_arr += $line
}

## Create Powershell object to tally Robocopy results
$row = "" |select COPIED, MISMATCH, FAILED, EXTRAS
$row.COPIED = [int](($robo_arr[1] -split "\s+")[4]) + [int](($robo_arr[2] -split "\s+")[4])
$row.MISMATCH = [int](($robo_arr[1] -split "\s+")[6]) + [int](($robo_arr[2] -split "\s+")[6])
$row.FAILED = [int](($robo_arr[1] -split "\s+")[7]) + [int](($robo_arr[2] -split "\s+")[7])
$row.EXTRAS = [int](($robo_arr[1] -split "\s+")[8]) + [int](($robo_arr[2] -split "\s+")[8])

## If there are differences, lets kick off robocopy again
if ( ($row.COPIED + $row.MISMATCH + $row.FAILED + $row.EXTRAS) -gt 0 ){
    robocopy "$cSrc" "$cDst" /MIR /FFT /Z /W:5 /MT:64 /XX /log:$cLog
    Invoke-item $cLog
}
else { Write-host "Folders '$cSrc' and '$cDst' are twins" }

Примечание: я изменил приведенное выше регулярное выражение с точки на пробел непосредственно перед звездочкой. Это исключает риск отлова "файлов", "папок" или "итогов" в имени файла / пути. Оригинальная линия ниже для потомков.
JSK 2014/07/16

$ robo_results = $ robo_test -match '^ (? =.? \ b (Total | Dirs | Files) \ b) ((?! Files).)$'

Я понимаю, что это более старый поток, но учтите, что это также может быть выполнено с использованием кода завершения, возвращаемого robocopy, чтобы определить, существуют ли какие-либо изменения (значение 0 указывает на отсутствие изменений). Есть 2 способа сделать это:

  1. Запустите оператор robocopy выше, а затем в следующей строке проверьте значение $LastExitCode.
  2. Используйте Start-Process:
$RobocopyResult = Запуск-Процесс -FilePath robocopy -ArgumentList $RobocopyParams -Wait -PassThru
$RobocopyResult.ExitCode

При использовании Start-Process важно использовать -PassThru, иначе в $RobocopyResult не будет установлено значение.

Другие коды выхода перечислены по адресу http://ss64.com/nt/robocopy-exit.html.

Вот мой взлом на это. Мой код читает файл журналов и ищет строки, соответствующие полям сводки. Затем он разбирает его на два объекта, которые содержат статистику для файлов и каталогов:

        ### Read data from log summary at end of RoboCopy log
        $Props = ((cat $RoboCopyLogFile | ? {$_ -like "*Total*Copied*Skipped*Mismatch*FAILED*Extras*"}) | select -Last 1).split(" ") | ? {$_ -match "a|b|c|d|e|f|g|h|i|j|k|l|m|n|o|p|q|r|s|t|u|v|w|x|y|z"}
        $Dirs = (((cat $RoboCopyLogFile | ? {$_ -like "*Dirs : *"})) | select -Last 1).split(" ") | ? {$_ -match "1|2|3|4|5|6|7|8|9|0"}
        $Files = (((cat $RoboCopyLogFile | ? {$_ -like "*Files : *"})) | select -Last 1).split(" ") | ? {$_ -match "1|2|3|4|5|6|7|8|9|0"}
        ### Loop through the propteries of the summary
        $i = 0
        $FileStats = @{}
        $DirStats = @{}
        foreach($Prop in $Props)
        {
            $FileStats.("$Prop") = $Files[$i]
            $DirStats.("$Prop") = $Dirs[$i]
            $i++
        }
        $FileStats = New-Object -TypeName psobject -Property $FileStats
        $DirStats = New-Object -TypeName psobject -Property $DirStats
Другие вопросы по тегам