Como obter o robocopy em execução no powershell?

22

Estou tentando usar o robocopy dentro do powershell para espelhar alguns diretórios em minhas máquinas domésticas. Aqui está meu script:

param ($configFile)

$config = Import-Csv $configFile
$what = "/COPYALL /B /SEC/ /MIR"
$options = "/R:0 /W:0 /NFL /NDL"
$logDir = "C:\Backup\"

foreach ($line in $config)
{
 $source = $($line.SourceFolder)
 $dest = $($line.DestFolder)
 $logfile =  $logDIr 
 $logfile += Split-Path $dest -Leaf
 $logfile += ".log"

 robocopy "$source $dest $what $options /LOG:MyLogfile.txt"
}

O script recebe um arquivo csv com uma lista de diretórios de origem e destino. Quando executo o script, recebo esses erros:

-------------------------------------------------------------------------------
   ROBOCOPY     ::     Robust File Copy for Windows                              
-------------------------------------------------------------------------------

  Started : Sat Apr 03 21:26:57 2010

   Source : P:\ C:\Backup\Photos \COPYALL \B \SEC\ \MIR \R:0 \W:0 \NFL \NDL \LOG:MyLogfile.txt\
     Dest - 

    Files : *.*

  Options : *.* /COPY:DAT /R:1000000 /W:30 

------------------------------------------------------------------------------

ERROR : No Destination Directory Specified.

       Simple Usage :: ROBOCOPY source destination /MIR

             source :: Source Directory (drive:\path or \server\share\path).
        destination :: Destination Dir  (drive:\path or \server\share\path).
               /MIR :: Mirror a complete directory tree.

    For more usage information run ROBOCOPY /?


****  /MIR can DELETE files as well as copy them !

Alguma ideia do que preciso fazer para corrigir?

Obrigado Mark.

    
por Mark Allison 03.04.2010 / 21:29

8 respostas

10

Se as variáveis para $what e $options não mudarem, por que elas são variáveis?

$what = "/COPYALL /B /SEC /MIR" 
$options = "/R:0 /W:0 /NFL /NDL" 

Isso funciona bem para mim:

robocopy   $source $dest /COPYALL /B /SEC /MIR /R:0 /W:0 /NFL /NDL
    
por 04.04.2010 / 01:32
19

Preencher sequências de caracteres em parâmetros para comandos externos do Powershell requer alguns saltos se você quiser usar a expansão de variáveis e também a linha de comando resultante entender corretamente quais parâmetros você deseja separar e quais não devem. No seu exemplo, você está enviando a string inteira como o primeiro parâmetro e o Robocopy está dizendo que não pode encontrar essa longa string como um diretório de origem. Você quer que toda a string Source seja tratada como um parâmetro (incluindo espaços que podem estar lá), da mesma forma para Destination, mas suas opções $ what e $ falharão porque ambas serão entregues ao Robocopy como um único parâmetro que não pode ser analisado.

Existem algumas maneiras de fazer isso direito, mas o snippet a seguir é baseado na maneira como você parece querer quebrar seus parâmetros e funciona para o exemplo de diretório único que usei.

$source="C:\temp\source"
$dest="C:\temp\dest"

$what = @("/COPYALL","/B","/SEC","/MIR")
$options = @("/R:0","/W:0","/NFL","/NDL")

$cmdArgs = @("$source","$dest",$what,$options)
robocopy @cmdArgs
    
por 03.04.2010 / 23:38
3

Remova as aspas da linha de robocopy?

ou seja,

param ($configFile)

$config = Import-Csv $configFile
$what = "/COPYALL /B /SEC/ /MIR"
$options = "/R:0 /W:0 /NFL /NDL"
$logDir = "C:\Backup\"

foreach ($line in $config)
{
 $source = $($line.SourceFolder)
 $dest = $($line.DestFolder)
 $logfile =  $logDIr 
 $logfile += Split-Path $dest -Leaf
 $logfile += ".log"

 robocopy $source $dest $what $options /LOG:MyLogfile.txt
}
    
por 03.04.2010 / 21:46
2

Eu tive o mesmo problema. A linha de comando inteira foi interpretada como o primeiro argumento.

Nada mencionado acima funcionou, incluindo:

invoke-expression "robocopy ""$sourceFolder"" ""$destinationFolder"" /MIR"  
invoke-expression "robocopy \'"$sourceFolder\'" \'"$destinationFolder\'" /MIR"  

Deixar as cotações não funciona quando há um espaço no caminho:

invoke-expression "robocopy $sourceFolder $destinationFolder /MIR"  

Depois de tentar os truques no link , finalmente consegui.

robocopy "\'"$sourceFolder"\'" "\'"$destinationFolder"\'" /MIR

Talvez eu devesse ter ficado com arquivos bat. :)

    
por 16.07.2011 / 23:06
1

Descobri que a melhor maneira de executar um comando nativo é usar o & comando para executar uma lista de strings. Além disso, se você quiser que a saída do console seja incluída em uma transcrição do PowerShell, será necessário enviar a saída para fora do host. Aqui está um exemplo de chamar o 7Zip com vários parâmetros retirados de um 7-Zip para o Amazon S3 Powershell Script que escrevi:

$7ZipPath = "C:\Program Files-Zipz.exe" #Path to 7Zip executable
$FolderPath = "C:\Backup" #Folder to backup (no trailing slash!)
$FolderName = $FolderPath.Substring($FolderPath.LastIndexOf("\")+1) #grab the name of the backup folder
$TimeStamp = [datetime]::Now.ToString("yyyy-MM-dd_HHmm") #Create unique timestamp string
$strFile = [String]::Format("{0}\{1}_{2}.7z", "c:\",$FolderName,$TimeStamp) #Create filename for the zip
#7z options: add, type 7z, Archive filename, Files to add (with wildcard. Change \* to \prefix* or \*.txt to limit files), compression level 9, password, encrypt filenames, Send output to host so it can be logged.
& $7ZipPath "a" "-t7z" "$strFile" "$FolderPath\*" "-mx9" "-r" "-pPASSWORD" "-mhe" | out-host #create archive file
if($LASTEXITCODE -ne 0){ #Detect errors from 7Zip. Note: 7z will crash sometimes if file already exists
    Write-Error "ERROR: 7Zip terminated with exit code $LASTEXITCODE. Script halted."
    exit $LASTEXITCODE
}

Basicamente, para o seu caso, eu tentaria algo assim (talvez seja necessário quebrar os parâmetros $ what e $ options individualmente):

$robocopypath = "c:\pathtofolder\robocopy.exe"
& $robocopypath "$source" "$dest" "$what" "$options" "/LOG:MyLogfile.txt"
    
por 04.04.2010 / 21:08
0
invoke-expression "robocopy $source $dest $what $options /LOG:MyLogfile.txt"

Isso irá interpolar todas as variáveis e, em seguida, chamar a string resultante. Do jeito que você tem agora, parece que o robocopy está sendo invocado com as aspas em torno de todas as opções, ou seja, o robocopy "c: \ src c: \ dst blah meh". Isso é interpretado como um único parâmetro.

    
por 01.05.2010 / 15:42
0

Isso funcionou para mim e permite uma entrada dinâmica do local do arquivo de log. o & símbolo apenas diz ao PowerShell que o texto é uma linha de código que eu quero executar.

    $source = "E:\BackupToRestore\"
    $destination = "E:\RestoreMe\"
    $logfile = "E:\robocopyLogFile.txt"    
    $switches = ("/B", "/MIR", "/XJ", "/FFT", "/R:0", "/LOG:$logfile")
    & robocopy $source $destination $roboCopyString $switches
    
por 22.10.2015 / 19:23
0

Adicionando meus 2 centavos para isso, pois pode ajudar alguém ... O código abaixo está faltando a parte "loop", onde eu li um csv para obter os arquivos para copiar

# first we setup an array of possible robocopy status
$RobocopyErrors="NO ERRORS",
            "OKCOPY",
            "XTRA",
            "OKCOPY + XTRA",
            "MISMATCHES",
            "OKCOPY + MISMATCHES",
            "MISMATCHES + XTRA",
            "OKCOPY + MISMATCHES + XTRA",
            "FAIL",
            "OKCOPY + FAIL",
            "FAIL + XTRA",
            "OKCOPY + FAIL + XTRA",
            "FAIL + MISMATCHES& goto end",
            "OKCOPY + FAIL + MISMATCHES",
            "FAIL + MISMATCHES + XTRA",
            "OKCOPY + FAIL + MISMATCHES + XTRA",
            "***FATAL ERROR***"
#setting some variables with the date
$DateLogFile=get-date -format "yyyyddMMhh"

#this commands below one usually goes into a loop
$DateLog=get-date -format "yyyyddMMhhmmss"

#adding options and command arg as described in previous post
$options = @("/R:0","/W:0")
$cmdArgs = @("$Source","$Destination",$File,$options)

#executing Robocopy command
robocopy @cmdArgs

# We capture the lastexitcode of the command and use to confirm if all was good or not

$errorlevel=$LASTEXITCODE
# I output the status to a log file
"$DateLog't::$($RobocopyErrors[$errorlevel])::'"$file'"'t'"$Source'" -> '"$Destination'"" | out-file "$scriptPath\Logs\$DateLogFile.log" -append
 if ($errorlevel -lt 8) {
    #error below 8 = all is good

}
else {
    # something bad happened ..  

}
    
por 18.04.2018 / 09:53