Powershell 2 copy-item, который создает папку, если ее не существует

$from = "\\something\1 XLS\2010_04_22\*"
$to =  "c:\out\1 XLS\2010_04_22\"
copy-item $from $to -Recurse 

Это работает, если c:\out\1 XLS\2010_04_22\ существует. Можно ли с помощью одной команды создать c:\out\1 XLS\2010_04_22\ если он не существует?

Ответ 1

Да, добавьте параметр -Force.

copy-item $from $to -Recurse -Force

Ответ 2

В PowerShell 2.0 по-прежнему невозможно получить командлет Copy-Item для создания папки назначения, вам понадобится следующий код:

$destinationFolder = "C:\My Stuff\Subdir"

if (!(Test-Path -path $destinationFolder)) {New-Item $destinationFolder -Type Directory}
Copy-Item "\\server1\Upgrade.exe" -Destination $destinationFolder

Если вы используете -Recurse в Copy-Item, он создаст все вложенные папки исходной структуры в получателе, но не создаст фактическую папку назначения даже с помощью параметра -Force.

Ответ 3

В PowerShell 3 и выше я использую Copy-Item with New-Item.

copy-item -Path $file -Destination (new-item -type directory -force ("C:\Folder\sub\sub\" + $newSub)) -force -ea 0

Я не пробовал это в версии 2.

Ответ 4

  $filelist | % {
    $file = $_
    mkdir -force (Split-Path $dest) | Out-Null
    cp $file $dest
  } 

Ответ 5

function Copy-File ([System.String] $sourceFile, [System.String] $destinationFile, [Switch] $overWrite) {

    if ($sourceFile -notlike "filesystem::*") {
        $sourceFile = "filesystem::$sourceFile" 
    }

    if ($destinationFile -notlike "filesystem::*") {
        $destinationFile = "filesystem::$destinationFile" 
    }

    $destinationFolder = $destinationFile.Replace($destinationFile.Split("\")[-1],"")

    if (!(Test-Path -path $destinationFolder)) {
        New-Item $destinationFolder -Type Directory
    }

    try {
        Copy-Item -Path $sourceFile -Destination $destinationFile -Recurse -Force
        Return $true 
    } catch [System.IO.IOException] {
        # If overwrite enabled, then delete the item from the destination, and try again:
        if ($overWrite) {
            try {
                Remove-Item -Path $destinationFile -Recurse -Force        
                Copy-Item -Path $sourceFile -Destination $destinationFile -Recurse -Force 
                Return $true
            } catch {
                Write-Error -Message "[Copy-File] Overwrite error occurred!`n$_" -ErrorAction SilentlyContinue
                #$PSCmdlet.WriteError($Global:Error[0])
                Return $false
            }
        } else {
            Write-Error -Message "[Copy-File] File already exists!" -ErrorAction SilentlyContinue
            #$PSCmdlet.WriteError($Global:Error[0])
            Return $false
        }
    } catch {
        Write-Error -Message "[Copy-File] File move failed!`n$_" -ErrorAction SilentlyContinue
        #$PSCmdlet.WriteError($Global:Error[0]) 
        Return $false
    } 
}

Ответ 6

Моим любимым является использование класса .Net [IO.DirectoryInfo], который позаботится о некоторой логике. Я действительно использую это для множества подобных задач сценариев. Он имеет метод .Create(), который создает каталоги, которые не существуют, без ошибок, если они это делают.

Поскольку это еще проблема с двумя шагами, я использую псевдоним foreach, чтобы он был простым. Для отдельных файлов:

[IO.DirectoryInfo]$to |% {$_.create(); cp $from $_}

Что касается вашего файла с несколькими файлами/каталогами, я бы использовал RoboCopy над xcopy. Удалите "*" из своего и просто используйте:

RoboCopy.exe $from $to *

Вы все равно можете добавить /r (Recurse),/e (Recurse, включая Empty), а также 50 других полезных переключателей.

Изменить: Оглядываясь на это, он краток, но не очень читабельен, если вы часто не используете код. Обычно я разбил его на две части, например:

([IO.DirectoryInfo]$to).Create()
cp $from $to

Кроме того, DirectoryInfo является типом родительского свойства FileInfo, поэтому, если ваш $to является файлом, вы можете использовать их вместе:

([IO.FileInfo]$to).Parent.Create()
cp $from $to

Ответ 7

Вот пример, который работал для меня. У меня был список из 500 конкретных файлов в текстовом файле, содержащий около 100 различных папок, которые я должен был скопировать в резервную копию на случай, если эти файлы понадобятся позже. Текстовый файл содержал полный путь и имя файла, по одному на строку. В моем случае я хотел убрать букву диска и имя первой подпапки из каждого имени файла. Я хотел скопировать все эти файлы в похожую структуру папок в другой корневой папке назначения, которую я указал. Я надеюсь, что другие пользователи найдут это полезным.

# Copy list of files (full path + file name) in a txt file to a new destination, creating folder structure for each file before copy
$rootDestFolder = "F:\DestinationFolderName"
$sourceFiles = Get-Content C:\temp\filelist.txt
foreach($sourceFile in $sourceFiles){
    $filesplit = $sourceFile.split("\")
    $splitcount = $filesplit.count
    # This example strips the drive letter & first folder ( ex: E:\Subfolder\ ) but appends the rest of the path to the rootDestFolder
    $destFile = $rootDestFolder + "\" + $($($sourceFile.split("\")[2..$splitcount]) -join "\")
    # Output List of source and dest 
    Write-Host ""
    Write-Host "===$sourceFile===" -ForegroundColor Green
    Write-Host "+++$destFile+++"
    # Create path and file, if they do not already exist
    $destPath = Split-Path $destFile
    If(!(Test-Path $destPath)) { New-Item $destPath -Type Directory }
    If(!(Test-Path $destFile)) { Copy-Item $sourceFile $destFile }
}

Ответ 8

Я дважды наткнулся на это, и этот последний раз был уникальной ситуацией, и даже если я использую copy-item, я хотел опубликовать решение, которое использовал.

Имеет список только файлов с полным путем, и в большинстве случаев файлы не имеют расширений. параметр -Recurse -Force не будет работать для меня, поэтому я отбросил функцию copy-item и вернулся к чему-то вроде ниже, используя xcopy, поскольку я все еще хотел сохранить его одним лайнером. Первоначально я был связан с Robocopy, но он, по-видимому, ищет расширение файла, и поскольку многие из моих не имели расширения, он считал его каталогом.

$filelist = @("C:\Somepath\test\location\here\file","C:\Somepath\test\location\here2\file2")

$filelist | % { echo f | xcopy $_  $($_.Replace("somepath", "somepath_NEW")) }

Надеюсь, это поможет кому-то.