Skopiuj całą zawartość katalogu do innego za pomocą php

146

Próbowałem skopiować całą zawartość katalogu do innej lokalizacji za pomocą

copy ("old_location/*.*","new_location/");

ale mówi, że nie może znaleźć strumienia, prawda *.*nie została znaleziona.

Każdy inny sposób

Dzięki, Dave

dave
źródło
1
@ redaktorzy: Czy na pewno to "old_location/."była tylko literówka?
Felix Kling
Rich Rodecker ma na swoim blogu skrypt, który wydaje się właśnie to robić. visible-form.com/blog/copy-directory-in-php
Jon F Hancock
@Felix: Zastanawiałem się nad tym samym. Cofnąłem się do pierwszej wersji, ale tak się stało "old_location/*.*. Nie mogę znaleźć wersji zawierającej "old_location/.".
Asaph
@Asaph: Twoje wycofanie było w porządku, spójrz na historię ... Miałem na myślicopy ("old_location/.","new_location/");
Felix Kling
3
@dave Kiedy odbierzesz zaakceptowane :)?
Nam G VU

Odpowiedzi:

239

Wygląda na to, że kopia obsługuje tylko pojedyncze pliki . Oto funkcja rekurencyjnego kopiowania, którą znalazłem w tej uwadze na stronie dokumentacji kopiowania :

<?php 
function recurse_copy($src,$dst) { 
    $dir = opendir($src); 
    @mkdir($dst); 
    while(false !== ( $file = readdir($dir)) ) { 
        if (( $file != '.' ) && ( $file != '..' )) { 
            if ( is_dir($src . '/' . $file) ) { 
                recurse_copy($src . '/' . $file,$dst . '/' . $file); 
            } 
            else { 
                copy($src . '/' . $file,$dst . '/' . $file); 
            } 
        } 
    } 
    closedir($dir); 
} 
?>
Felix Kling
źródło
2
To gwiazdka, a nie gwiazda;)
Gordon
6
Działa jak urok ... Dzięki @FelixKling
Milap
2
Dlaczego @mkdirzamiast mkdir?
Oliboy50,
3
@ Oliboy50: Możesz zapytać osobę, która napisała kod 5 lat temu: php.net/manual/en/function.copy.php#91010 . Może wtedy bardziej popularne było tłumienie komunikatów o błędach.
Felix Kling,
1
@ Oliboy50: Rozumiem. Myślę, że usuwa wszelkie komunikaty o błędach. Jednak nigdy go nie użyłem. Oto dokumentacja: us3.php.net/manual/en/language.operators.errorcontrol.php
Felix Kling.
90

Jak opisano tutaj , jest to kolejne podejście, które zajmuje się również linkami symbolicznymi:

/**
 * Copy a file, or recursively copy a folder and its contents
 * @author      Aidan Lister <[email protected]>
 * @version     1.0.1
 * @link        http://aidanlister.com/2004/04/recursively-copying-directories-in-php/
 * @param       string   $source    Source path
 * @param       string   $dest      Destination path
 * @param       int      $permissions New folder creation permissions
 * @return      bool     Returns true on success, false on failure
 */
function xcopy($source, $dest, $permissions = 0755)
{
    $sourceHash = hashDirectory($source);
    // Check for symlinks
    if (is_link($source)) {
        return symlink(readlink($source), $dest);
    }

    // Simple copy for a file
    if (is_file($source)) {
        return copy($source, $dest);
    }

    // Make destination directory
    if (!is_dir($dest)) {
        mkdir($dest, $permissions);
    }

    // Loop through the folder
    $dir = dir($source);
    while (false !== $entry = $dir->read()) {
        // Skip pointers
        if ($entry == '.' || $entry == '..') {
            continue;
        }

        // Deep copy directories
        if($sourceHash != hashDirectory($source."/".$entry)){
             xcopy("$source/$entry", "$dest/$entry", $permissions);
        }
    }

    // Clean up
    $dir->close();
    return true;
}

// In case of coping a directory inside itself, there is a need to hash check the directory otherwise and infinite loop of coping is generated

function hashDirectory($directory){
    if (! is_dir($directory)){ return false; }

    $files = array();
    $dir = dir($directory);

    while (false !== ($file = $dir->read())){
        if ($file != '.' and $file != '..') {
            if (is_dir($directory . '/' . $file)) { $files[] = hashDirectory($directory . '/' . $file); }
            else { $files[] = md5_file($directory . '/' . $file); }
        }
    }

    $dir->close();

    return md5(implode('', $files));
}
itsjavi
źródło
Świetnie się spisało, aby skopiować folder ze 140 podfolderami i każdym podfolderem zawierającym 21 obrazów. Działa świetnie! Dzięki!
Darksaint2014
1
mkdirpowinien zostać dodany truejako ostatni parametr do obsługi katalogu rekurencyjnego, wtedy ten skrypt jest doskonały
ZenithS
Spowoduje to skopiowanie całego folderu? A co, jeśli chcesz skopiować tylko pliki wewnątrz folderu, bez folderu nadrzędnego, zgodnie z pytaniem: copy ("old_location/*.*","new_location/");Czy to działa? A jeśli masz pliki z kropkami, czy zostaną dopasowane?
XCS
35

copy () działa tylko z plikami.

Zarówno polecenia copy systemu DOS, jak i polecenia cp w systemie Unix będą kopiować rekursywnie - więc najszybszym rozwiązaniem jest po prostu ich wyrzucenie i użycie. na przykład

`cp -r $src $dest`;

W przeciwnym razie będziesz musiał użyć opendir/ readdirlub scandirdo odczytania zawartości katalogu, iteracji w wynikach i jeśli is_dir zwróci true dla każdego z nich, powtórz w nim.

na przykład

function xcopy($src, $dest) {
    foreach (scandir($src) as $file) {
        if (!is_readable($src . '/' . $file)) continue;
        if (is_dir($src .'/' . $file) && ($file != '.') && ($file != '..') ) {
            mkdir($dest . '/' . $file);
            xcopy($src . '/' . $file, $dest . '/' . $file);
        } else {
            copy($src . '/' . $file, $dest . '/' . $file);
        }
    }
}
symcbean
źródło
1
Oto bardziej stabilna i czystsza wersja xcopy (), która nie tworzy folderu, jeśli istnieje: function xcopy($src, $dest) { foreach (scandir($src) as $file) { $srcfile = rtrim($src, '/') .'/'. $file; $destfile = rtrim($dest, '/') .'/'. $file; if (!is_readable($srcfile)) { continue; } if ($file != '.' && $file != '..') { if (is_dir($srcfile)) { if (!file_exists($destfile)) { mkdir($destfile); } xcopy($srcfile, $destfile); } else { copy($srcfile, $destfile); } } } }
TheStoryCoder
Dzięki za rozwiązanie backtick ! Strona, która pomogła mi ulepszyć polecenie kopiowania: wyjaśniono UNIX cp . Dodatkowe informacje: PHP> = 5.3 oferuje kilka fajnych iteratorów
maxpower9000
21

Najlepszym rozwiązaniem jest!

<?php
$src = "/home/www/domain-name.com/source/folders/123456";
$dest = "/home/www/domain-name.com/test/123456";

shell_exec("cp -r $src $dest");

echo "<H3>Copy Paste completed!</H3>"; //output when done
?>
czarny
źródło
31
Nie będzie działać na serwerach Windows ani w innych środowiskach, w których nie masz dostępu do shell_execlub cp. To sprawia, że ​​- moim zdaniem - nie jest to „najlepsze” rozwiązanie.
The Pellmeister
3
Poza tym sterowanie wierszem poleceń z pliku PHP może być dużym problemem, gdy ktoś znajdzie sposób na pobranie pliku na serwer.
Martijn,
Działał jak urok! Na CentOS i działało świetnie. Dzięki @bstpierre
Nick Green
1
To w ogóle nie zadziała w systemie Windows, ponieważ cpjest to polecenie systemu Linux. Do użytku w systemie Windows xcopy dir1 dir2 /e /i, gdzie /eoznacza kopiowanie pustych katalogów i /iignorowanie pytań dotyczących plików lub katalogów
Michel
Tak, jest to najlepsze rozwiązanie, jeśli Twój serwer obsługuje to polecenie i masz wymagane uprawnienia. To jest bardzo szybkie. Niestety nie działa we wszystkich środowiskach.
mdikici
13
function full_copy( $source, $target ) {
    if ( is_dir( $source ) ) {
        @mkdir( $target );
        $d = dir( $source );
        while ( FALSE !== ( $entry = $d->read() ) ) {
            if ( $entry == '.' || $entry == '..' ) {
                continue;
            }
            $Entry = $source . '/' . $entry; 
            if ( is_dir( $Entry ) ) {
                full_copy( $Entry, $target . '/' . $entry );
                continue;
            }
            copy( $Entry, $target . '/' . $entry );
        }

        $d->close();
    }else {
        copy( $source, $target );
    }
}
kzoty
źródło
Działa świetnie! Dzięki bracie
Robin Delaporte
8

Jak wspomniano w innym miejscu, copydziała tylko z jednym plikiem źródłowym, a nie wzorcem. Jeśli chcesz kopiować według wzorca, użyj globdo określenia plików, a następnie uruchom kopiowanie. Nie spowoduje to jednak skopiowania podkatalogów ani utworzenia katalogu docelowego.

function copyToDir($pattern, $dir)
{
    foreach (glob($pattern) as $file) {
        if(!is_dir($file) && is_readable($file)) {
            $dest = realpath($dir . DIRECTORY_SEPARATOR) . basename($file);
            copy($file, $dest);
        }
    }    
}
copyToDir('./test/foo/*.txt', './test/bar'); // copies all txt files
Gordon
źródło
rozważ zmianę: $ dest = realpath ($ dir. DIRECTORY_SEPARATOR). basename ($ plik); gdzie: $ dest = realpath ($ dir). DIRECTORY_SEPARATOR. basename ($ plik);
dawez
8
<?php
    function copy_directory( $source, $destination ) {
        if ( is_dir( $source ) ) {
        @mkdir( $destination );
        $directory = dir( $source );
        while ( FALSE !== ( $readdirectory = $directory->read() ) ) {
            if ( $readdirectory == '.' || $readdirectory == '..' ) {
                continue;
            }
            $PathDir = $source . '/' . $readdirectory; 
            if ( is_dir( $PathDir ) ) {
                copy_directory( $PathDir, $destination . '/' . $readdirectory );
                continue;
            }
            copy( $PathDir, $destination . '/' . $readdirectory );
        }

        $directory->close();
        }else {
        copy( $source, $destination );
        }
    }
?>

z ostatniej czwartej linii, zrób to

$source = 'wordpress';//i.e. your source path

i

$destination ='b';
Hemanta Nandi
źródło
7

Pełne podziękowania należą się Felixowi Klingowi za doskonałą odpowiedź, której z wdzięcznością użyłem w moim kodzie. Oferuję niewielkie rozszerzenie wartości logicznej zwracanej w celu zgłaszania sukcesu lub niepowodzenia:

function recurse_copy($src, $dst) {

  $dir = opendir($src);
  $result = ($dir === false ? false : true);

  if ($result !== false) {
    $result = @mkdir($dst);

    if ($result === true) {
      while(false !== ( $file = readdir($dir)) ) { 
        if (( $file != '.' ) && ( $file != '..' ) && $result) { 
          if ( is_dir($src . '/' . $file) ) { 
            $result = recurse_copy($src . '/' . $file,$dst . '/' . $file); 
          }     else { 
            $result = copy($src . '/' . $file,$dst . '/' . $file); 
          } 
        } 
      } 
      closedir($dir);
    }
  }

  return $result;
}
gonzo
źródło
1
używasz recurse_copy () i recurseCopy () jako nazw funkcji, zaktualizuj je.
AgelessEssence
The closedir ($ dir); instrukcja musi znajdować się poza instrukcją if ($ reslut === true), tj. o jeden nawias klamrowy dalej. W przeciwnym razie ryzykujesz posiadaniem niezwolnionych zasobów.
Dimitar Darazhanski
5

Moja przycięta wersja odpowiedzi @Kzoty. Dziękuję Kzoty.

Stosowanie

Helper::copy($sourcePath, $targetPath);

class Helper {

    static function copy($source, $target) {
        if (!is_dir($source)) {//it is a file, do a normal copy
            copy($source, $target);
            return;
        }

        //it is a folder, copy its files & sub-folders
        @mkdir($target);
        $d = dir($source);
        $navFolders = array('.', '..');
        while (false !== ($fileEntry=$d->read() )) {//copy one by one
            //skip if it is navigation folder . or ..
            if (in_array($fileEntry, $navFolders) ) {
                continue;
            }

            //do copy
            $s = "$source/$fileEntry";
            $t = "$target/$fileEntry";
            self::copy($s, $t);
        }
        $d->close();
    }

}
Nam G VU
źródło
1

Klonuję cały katalog przez Iterator katalogu SPL.

function recursiveCopy($source, $destination)
{
    if (!file_exists($destination)) {
        mkdir($destination);
    }

    $splFileInfoArr = new RecursiveIteratorIterator(new RecursiveDirectoryIterator($source), RecursiveIteratorIterator::SELF_FIRST);

    foreach ($splFileInfoArr as $fullPath => $splFileinfo) {
        //skip . ..
        if (in_array($splFileinfo->getBasename(), [".", ".."])) {
            continue;
        }
        //get relative path of source file or folder
        $path = str_replace($source, "", $splFileinfo->getPathname());

        if ($splFileinfo->isDir()) {
            mkdir($destination . "/" . $path);
        } else {
        copy($fullPath, $destination . "/" . $path);
        }
    }
}
#calling the function
recursiveCopy(__DIR__ . "/source", __DIR__ . "/destination");
Tuhin Bepari
źródło
0
// using exec

function rCopy($directory, $destination)
{

    $command = sprintf('cp -r %s/* %s', $directory, $destination);

    exec($command);

}
Łukasz Szpak
źródło
0

W przypadku serwerów Linux potrzebujesz tylko jednej linii kodu do rekurencyjnego kopiowania z zachowaniem uprawnień:

exec('cp -a '.$source.' '.$dest);

Innym sposobem na to jest:

mkdir($dest);
foreach ($iterator = new \RecursiveIteratorIterator(new \RecursiveDirectoryIterator($source, \RecursiveDirectoryIterator::SKIP_DOTS), \RecursiveIteratorIterator::SELF_FIRST) as $item)
{
    if ($item->isDir())
        mkdir($dest.DIRECTORY_SEPARATOR.$iterator->getSubPathName());
    else
        copy($item, $dest.DIRECTORY_SEPARATOR.$iterator->getSubPathName());
}

ale jest wolniejszy i nie zachowuje uprawnień.

Dan Bray
źródło
0

Miałem podobną sytuację, w której musiałem kopiować z jednej domeny do drugiej na tym samym serwerze.Oto dokładnie to, co zadziałało w moim przypadku, możesz również dostosować do swojej:

foreach(glob('../folder/*.php') as $file) {
$adjust = substr($file,3);
copy($file, '/home/user/abcde.com/'.$adjust);

Zwróć uwagę na użycie „substr ()”, bez niej miejscem docelowym staje się „/home/user/abcde.com/../folder/”, co może być czymś, czego nie chcesz. Dlatego użyłem funkcji substr (), aby wyeliminować pierwsze 3 znaki (../), aby uzyskać żądane miejsce docelowe, którym jest „/home/user/abcde.com/folder/”. Możesz więc dostosować funkcję substr (), a także funkcję glob (), aż będzie pasować do twoich osobistych potrzeb. Mam nadzieję że to pomoże.

Chimdi
źródło