Прогресс во время большой копии файла (Копировать-элемент и запись-Прогресс?)
Есть ли способ скопировать действительно большой файл (с одного сервера на другой) в PowerShell и показать его ход?
Существуют решения для использования Write-Progress в сочетании с циклом для копирования множества файлов и отображения прогресса. Однако я не могу найти ничего, что показывало бы прогресс одного файла.
Какие-нибудь мысли?
Ответы
Ответ 1
Я не слышал о прогрессе с Copy-Item
. Если вы не хотите использовать какой-либо внешний инструмент, вы можете экспериментировать с потоками. Размер буфера меняется, вы можете попробовать разные значения (от 2 кб до 64 кб).
function Copy-File {
param( [string]$from, [string]$to)
$ffile = [io.file]::OpenRead($from)
$tofile = [io.file]::OpenWrite($to)
Write-Progress -Activity "Copying file" -status "$from -> $to" -PercentComplete 0
try {
[byte[]]$buff = new-object byte[] 4096
[int]$total = [int]$count = 0
do {
$count = $ffile.Read($buff, 0, $buff.Length)
$tofile.Write($buff, 0, $count)
$total += $count
if ($total % 1mb -eq 0) {
Write-Progress -Activity "Copying file" -status "$from -> $to" `
-PercentComplete ([int]($total/$ffile.Length* 100))
}
} while ($count -gt 0)
}
finally {
$ffile.Dispose()
$tofile.Dispose()
Write-Progress -Activity "Copying file" -Status "Ready" -Completed
}
}
Ответ 2
Кажется, что это гораздо лучшее решение для использования BitsTransfer, оно похоже на OOTB на большинстве машин Windows с PowerShell 2.0 или выше.
Import-Module BitsTransfer
Start-BitsTransfer -Source $Source -Destination $Destination -Description "Backup" -DisplayName "Backup"
Ответ 3
Альтернативно эта опция использует встроенный индикатор выполнения Windows...
$FOF_CREATEPROGRESSDLG = "&H0&"
$objShell = New-Object -ComObject "Shell.Application"
$objFolder = $objShell.NameSpace($DestLocation)
$objFolder.CopyHere($srcFile, $FOF_CREATEPROGRESSDLG)
Ответ 4
cmd /c copy /z src dest
не чистый PowerShell, а исполняемый в PowerShell и отображающий прогресс в процентах
Ответ 5
Я изменил код из stej (что было здорово, только то, что мне нужно!), чтобы использовать более крупный буфер, [long] для больших файлов и использовать класс System.Diagnostics.Stopwatch для отслеживания прошедшего времени и оценки оставшегося времени.
Также добавлено сообщение о скорости передачи во время передачи и вывода общего истекшего времени и общей скорости передачи.
Использование буфера 4 МБ (4096 * 1024 байт), чтобы стать лучше, чем собственное копирование пропускной способности Win7 с NAS на USB-накопитель на ноутбуке через Wi-Fi.
В списке дел:
- добавить обработку ошибок (catch)
- описать список файлов get-childitem в качестве входных данных
- вложенные индикаторы выполнения при копировании нескольких файлов (файл x из y,% if
копирование всех данных и т.д.)
- входной параметр для размера буфера
Не стесняйтесь использовать/улучшать: -)
function Copy-File {
param( [string]$from, [string]$to)
$ffile = [io.file]::OpenRead($from)
$tofile = [io.file]::OpenWrite($to)
Write-Progress `
-Activity "Copying file" `
-status ($from.Split("\")|select -last 1) `
-PercentComplete 0
try {
$sw = [System.Diagnostics.Stopwatch]::StartNew();
[byte[]]$buff = new-object byte[] (4096*1024)
[long]$total = [long]$count = 0
do {
$count = $ffile.Read($buff, 0, $buff.Length)
$tofile.Write($buff, 0, $count)
$total += $count
[int]$pctcomp = ([int]($total/$ffile.Length* 100));
[int]$secselapsed = [int]($sw.elapsedmilliseconds.ToString())/1000;
if ( $secselapsed -ne 0 ) {
[single]$xferrate = (($total/$secselapsed)/1mb);
} else {
[single]$xferrate = 0.0
}
if ($total % 1mb -eq 0) {
if($pctcomp -gt 0)`
{[int]$secsleft = ((($secselapsed/$pctcomp)* 100)-$secselapsed);
} else {
[int]$secsleft = 0};
Write-Progress `
-Activity ($pctcomp.ToString() + "% Copying file @ " + "{0:n2}" -f $xferrate + " MB/s")`
-status ($from.Split("\")|select -last 1) `
-PercentComplete $pctcomp `
-SecondsRemaining $secsleft;
}
} while ($count -gt 0)
$sw.Stop();
$sw.Reset();
}
finally {
write-host (($from.Split("\")|select -last 1) + `
" copied in " + $secselapsed + " seconds at " + `
"{0:n2}" -f [int](($ffile.length/$secselapsed)/1mb) + " MB/s.");
$ffile.Close();
$tofile.Close();
}
}
Ответ 6
Не то, чтобы я знал. В любом случае я бы не рекомендовал использовать экземпляр для этого. Я не думаю, что он был разработан так, чтобы быть надежным, как robocopy.exe, чтобы поддерживать повторную попытку, которую вы хотели бы использовать для чрезвычайно больших копий файлов по сети.
Ответ 7
Эта рекурсивная функция рекурсивно копирует файлы и каталоги из пути источника к пути назначения
Если файл уже существует в пути назначения, он копирует их только с более новыми файлами.
Function Copy-FilesBitsTransfer(
[Parameter(Mandatory=$true)][String]$sourcePath,
[Parameter(Mandatory=$true)][String]$destinationPath,
[Parameter(Mandatory=$false)][bool]$createRootDirectory = $true)
{
$item = Get-Item $sourcePath
$itemName = Split-Path $sourcePath -leaf
if (!$item.PSIsContainer){ #Item Is a file
$clientFileTime = Get-Item $sourcePath | select LastWriteTime -ExpandProperty LastWriteTime
if (!(Test-Path -Path $destinationPath\$itemName)){
Start-BitsTransfer -Source $sourcePath -Destination $destinationPath -Description "$sourcePath >> $destinationPath" -DisplayName "Copy Template file" -Confirm:$false
if (!$?){
return $false
}
}
else{
$serverFileTime = Get-Item $destinationPath\$itemName | select LastWriteTime -ExpandProperty LastWriteTime
if ($serverFileTime -lt $clientFileTime)
{
Start-BitsTransfer -Source $sourcePath -Destination $destinationPath -Description "$sourcePath >> $destinationPath" -DisplayName "Copy Template file" -Confirm:$false
if (!$?){
return $false
}
}
}
}
else{ #Item Is a directory
if ($createRootDirectory){
$destinationPath = "$destinationPath\$itemName"
if (!(Test-Path -Path $destinationPath -PathType Container)){
if (Test-Path -Path $destinationPath -PathType Leaf){ #In case item is a file, delete it.
Remove-Item -Path $destinationPath
}
New-Item -ItemType Directory $destinationPath | Out-Null
if (!$?){
return $false
}
}
}
Foreach ($fileOrDirectory in (Get-Item -Path "$sourcePath\*"))
{
$status = Copy-FilesBitsTransfer $fileOrDirectory $destinationPath $true
if (!$status){
return $false
}
}
}
return $true
}
Ответ 8
У Тревора Салливана есть описание того, как добавить команду под названием Copy-ItemWithProgress в PowerShell в Robocopy.
Ответ 9
Шон Кирни из Hey, автор сценариев! У блога есть решение, которое я нашел, работает довольно хорошо.
Function Copy-WithProgress
{
[CmdletBinding()]
Param
(
[Parameter(Mandatory=$true,
ValueFromPipelineByPropertyName=$true,
Position=0)]
$Source,
[Parameter(Mandatory=$true,
ValueFromPipelineByPropertyName=$true,
Position=0)]
$Destination
)
$Source=$Source.tolower()
$Filelist=Get-Childitem "$Source" –Recurse
$Total=$Filelist.count
$Position=0
foreach ($File in $Filelist)
{
$Filename=$File.Fullname.tolower().replace($Source,'')
$DestinationFile=($Destination+$Filename)
Write-Progress -Activity "Copying data from '$source' to '$Destination'" -Status "Copying File $Filename" -PercentComplete (($Position/$total)*100)
Copy-Item $File.FullName -Destination $DestinationFile
$Position++
}
}
Тогда использовать это:
Copy-WithProgress -Source $src -Destination $dest
Ответ 10
Ненавижу быть тем, кто поднимает старую тему, но я нашел этот пост чрезвычайно полезным. После выполнения тестов производительности на фрагментах от stej и его уточнения Грэмом Голдом, а также предложением Nacht от BITS я пришел к выводу, что:
- Мне очень понравилась команда Грэма с оценками времени и показаниями скорости.
- Мне также очень понравилось значительное увеличение скорости использования BITS в качестве метода передачи.
Столкнувшись с решением между двумя... Я обнаружил, что Start-BitsTransfer поддерживает асинхронный режим. Итак, вот результат моего слияния двух.
function Copy-File {
param([string]$from, [string]$to)
try {
$job = Start-BitsTransfer -Source $from -Destination $to '
-Description "Moving: $from => $to" '
-DisplayName "Backup" -Asynchronous
# Start stopwatch
$sw = [System.Diagnostics.Stopwatch]::StartNew()
Write-Progress -Activity "Connecting..."
while ($job.JobState.ToString() -ne "Transferred") {
switch ($job.JobState.ToString()) {
"Connecting" {
break
}
"Transferring" {
$pctcomp = ($job.BytesTransferred / $job.BytesTotal) * 100
$elapsed = ($sw.elapsedmilliseconds.ToString()) / 1000
if ($elapsed -eq 0) {
$xferrate = 0.0
} else {
$xferrate = (($job.BytesTransferred / $elapsed) / 1mb);
}
if ($job.BytesTransferred % 1mb -eq 0) {
if ($pctcomp -gt 0) {
$secsleft = ((($elapsed / $pctcomp) * 100) - $elapsed)
} else {
$secsleft = 0
}
Write-Progress -Activity ("Copying file '" + ($PathName.Split("\") | Select -last 1) + "' @ " + "{0:n2}" -f $xferrate + "MB/s") '
-PercentComplete $pctcomp '
-SecondsRemaining $secsleft
}
break
}
"Transferred" {
break
}
Default {
throw $job.JobState.ToString() + " unexpected BITS state."
}
}
}
$sw.Stop()
$sw.Reset()
} finally {
Complete-BitsTransfer -BitsJob $job
Write-Progress -Activity "Completed" -Completed
}
}