Як зафіксувати всю папку за допомогою PHP


131

Я знайшов тут у stackoveflow кілька кодів про те, як ZIP-файл певного файлу, а як щодо певної папки?

Folder/
  index.html
  picture.jpg
  important.txt

всередині My Folder, є файли. після блискавки My Folderя також хочу видалити весь вміст папки, за винятком important.txt.

Знайшли це тут у стеці

Мені потрібна твоя допомога. Дякую.


Наскільки я можу бачити, надане вами посилання stackoverflow насправді створює декілька файлів. З якою частиною у вас проблеми?
Lasse Espeholt

@lasseespeholt Посилання, яке я дав вам блискавки лише конкретний файл, а не папку та вміст папки ..
wininana

Він бере масив файлів (фактично папку) і додає всі файли до zip-файлу (циклу). Я бачу точну відповідь була розміщена зараз +1 :), це той самий код, масив - це лише список файлів із каталогу.
Lasse Espeholt


Це може допомогти вам codingbin.com/compressing-a-directory-of-files-with-php
MKD

Відповіді:


320

Код оновлено 2015/04/22.

Скопіюйте всю папку:

// Get real path for our folder
$rootPath = realpath('folder-to-zip');

// Initialize archive object
$zip = new ZipArchive();
$zip->open('file.zip', ZipArchive::CREATE | ZipArchive::OVERWRITE);

// Create recursive directory iterator
/** @var SplFileInfo[] $files */
$files = new RecursiveIteratorIterator(
    new RecursiveDirectoryIterator($rootPath),
    RecursiveIteratorIterator::LEAVES_ONLY
);

foreach ($files as $name => $file)
{
    // Skip directories (they would be added automatically)
    if (!$file->isDir())
    {
        // Get real and relative path for current file
        $filePath = $file->getRealPath();
        $relativePath = substr($filePath, strlen($rootPath) + 1);

        // Add current file to archive
        $zip->addFile($filePath, $relativePath);
    }
}

// Zip archive will be created only after closing object
$zip->close();

Скопіюйте всю папку + видаліть усі файли, окрім "важливого.txt":

// Get real path for our folder
$rootPath = realpath('folder-to-zip');

// Initialize archive object
$zip = new ZipArchive();
$zip->open('file.zip', ZipArchive::CREATE | ZipArchive::OVERWRITE);

// Initialize empty "delete list"
$filesToDelete = array();

// Create recursive directory iterator
/** @var SplFileInfo[] $files */
$files = new RecursiveIteratorIterator(
    new RecursiveDirectoryIterator($rootPath),
    RecursiveIteratorIterator::LEAVES_ONLY
);

foreach ($files as $name => $file)
{
    // Skip directories (they would be added automatically)
    if (!$file->isDir())
    {
        // Get real and relative path for current file
        $filePath = $file->getRealPath();
        $relativePath = substr($filePath, strlen($rootPath) + 1);

        // Add current file to archive
        $zip->addFile($filePath, $relativePath);

        // Add current file to "delete list"
        // delete it later cause ZipArchive create archive only after calling close function and ZipArchive lock files until archive created)
        if ($file->getFilename() != 'important.txt')
        {
            $filesToDelete[] = $filePath;
        }
    }
}

// Zip archive will be created only after closing object
$zip->close();

// Delete all files from "delete list"
foreach ($filesToDelete as $file)
{
    unlink($file);
}

2
Ви повинні встановити chmod (для запису) на dir (де знаходиться цей скрипт) на 777. Наприклад: Якщо сценарій розташований у /var/www/localhost/script.php, тоді вам потрібно встановити chmod 0777 на dir / var / www / localhost /.
Дадор

3
Видалення файлів перед викликом $zip->close()не вийде. Перевірте мою відповідь тут
hek2mgl

10
@alnassre, це вимога з питання: "Я також хочу видалити весь вміст папки, крім важливих.txt". Також я раджу завжди читати код перед його виконанням.
Дадор

1
@alnassre hahahaha ... вибачте :) ... hahaha
Ondrej Rafaj

1
@ nick-newman, так, для обчислення відсотків ви можете використовувати php.net/manual/ru/function.iterator-count.php + лічильник всередині циклу. Що стосується рівня стиснення - це не можливо з ZipArchive в даний момент: stackoverflow.com/questions/1833168 / ...
Dador

54

У класі ZipArchive є корисний недокументований метод: addGlob ();

$zipFile = "./testZip.zip";
$zipArchive = new ZipArchive();

if ($zipArchive->open($zipFile, (ZipArchive::CREATE | ZipArchive::OVERWRITE)) !== true)
    die("Failed to create archive\n");

$zipArchive->addGlob("./*.txt");
if ($zipArchive->status != ZIPARCHIVE::ER_OK)
    echo "Failed to write files to zip\n";

$zipArchive->close();

Зараз задокументовано за адресою: www.php.net/manual/en/ziparchive.addglob.php


2
@netcoder - переваги написання phpt для його тестування ... в основному, прочитайте джерело для класу ZipArchive і знайдіть його там .... є також незадокументований метод addPattern (), який приймає шаблон стилю regexp, але мені ніколи не вдавалося змусити це працювати (може бути помилка в класі)
Марк Бейкер

1
@kread - ви можете використовувати це з будь-яким списком файлів, який можна витягти за допомогою glob (), тому я вважаю це надзвичайно корисним з моменту його відкриття.
Марк Бейкер

@MarkBaker Я знаю, що цей коментар проходить через роки після того, як ви опублікували, я просто пробую тут удачу. Я також розмістив тут питання про блискавки. Я збираюся спробувати глобальний метод, який ви розмістили тут, але моя головна проблема полягає в тому, що я не можу використовувати addFromString і використовую addFile, який просто мовчить. Ви, можливо, маєте ідею, що може піти не так, або що я можу зробити неправильно?
Skytiger

@ user1032531 - останній рядок мого допису (відредагований 13 грудня 2013 р.) вказує саме на це із посиланням на сторінку документів
Марк Бейкер

6
Чи є addGlobрекурсивним?
Вінсент Пуарьє

20

Спробуйте це:

$zip = new ZipArchive;
$zip->open('myzip.zip', ZipArchive::CREATE);
foreach (glob("target_folder/*") as $file) {
    $zip->addFile($file);
    if ($file != 'target_folder/important.txt') unlink($file);
}
$zip->close();

Це не буде рецидивувати застебнення.


Він, звичайно, видаляє деякі файли в My folder, але у мене також є папка в папці, My folderяка дає мені помилку: Дозвіл відхилено від’єднанням папки зMy folder
wininana

@Stupefy: Спробуйте if (!is_dir($file) && $file != 'target_folder...')натомість. Або перевірте @kread відповідь, якщо ви хочете рецидивувати блиск , це найефективніший спосіб.
netcoder

Папка всередині My folderдосі не видаляється, але помилок більше немає.
wininana

Я також забув згадати, що я не створюю .zip файлів.
wininana

1
Видалення файлів перед викликом $zip->close()не вийде. Перевірте тут
hek2mgl

19

Я припускаю, що це працює на сервері, де додаток zip знаходиться в шляху пошуку. Має бути справедливим для всіх Unix-систем, і, мабуть, більшість серверів на основі Windows.

exec('zip -r archive.zip "My folder"');
unlink('My\ folder/index.html');
unlink('My\ folder/picture.jpg');

Після цього архів буде розташований у archive.zip. Майте на увазі, що пробіли у назвах файлів чи папок є частою причиною помилок, і їх слід уникати, де це можливо.


15

Я спробував з кодом нижче, і він працює. Код пояснюється сам собою, будь ласка, повідомте мене, якщо у вас є якісь питання.

<?php
class FlxZipArchive extends ZipArchive 
{
 public function addDir($location, $name) 
 {
       $this->addEmptyDir($name);
       $this->addDirDo($location, $name);
 } 
 private function addDirDo($location, $name) 
 {
    $name .= '/';
    $location .= '/';
    $dir = opendir ($location);
    while ($file = readdir($dir))
    {
        if ($file == '.' || $file == '..') continue;
        $do = (filetype( $location . $file) == 'dir') ? 'addDir' : 'addFile';
        $this->$do($location . $file, $name . $file);
    }
 } 
}
?>

<?php
$the_folder = '/path/to/folder/to/be/zipped';
$zip_file_name = '/path/to/zip/archive.zip';
$za = new FlxZipArchive;
$res = $za->open($zip_file_name, ZipArchive::CREATE);
if($res === TRUE) 
{
    $za->addDir($the_folder, basename($the_folder));
    $za->close();
}
else{
echo 'Could not create a zip archive';
}
?>

Відмінне рішення. Він також працює в laravel 5.5. дуже сподобалось це. (y)
Веб-майстер

1
Чудовий код! Чисто, просто та ідеально працює! ;) Мені здається найкращою відповіддю. Якщо це може комусь допомогти: я щойно додав ini_set('memory_limit', '512M');перед виконанням сценарію і ini_restore('memory_limit');в кінці. Необхідно було уникнути нестачі пам’яті у випадку важких папок (це папка, більша за 500 Мб).
Якопо Пейс

1
У моєму середовищі (PHP 7.3, Debian) створено архів ZIP без переліку каталогів (великий, порожній файл). Мені довелося змінити наступний рядок: $ name. = '/'; в $ name = ($ name == '.'? '': $ name. '/');
Герфрід

Це працює для мене. Дякую, що поділились. Ура!
Сатиська

8

Це функція, яка збирає всю папку та її вміст у zip-файл, і ви можете використовувати її просто так:

addzip ("path/folder/" , "/path2/folder.zip" );

функція:

// compress all files in the source directory to destination directory 
    function create_zip($files = array(), $dest = '', $overwrite = false) {
    if (file_exists($dest) && !$overwrite) {
        return false;
    }
    if (($files)) {
        $zip = new ZipArchive();
        if ($zip->open($dest, $overwrite ? ZIPARCHIVE::OVERWRITE : ZIPARCHIVE::CREATE) !== true) {
            return false;
        }
        foreach ($files as $file) {
            $zip->addFile($file, $file);
        }
        $zip->close();
        return file_exists($dest);
    } else {
        return false;
    }
}

function addzip($source, $destination) {
    $files_to_zip = glob($source . '/*');
    create_zip($files_to_zip, $destination);
    echo "done";
}

Як автоматично включати підпапки в резервну копію за допомогою цього сценарію? @Alireza
floCoder

2

Чому б не спробувати EFS PhP-ZiP MultiVolume Script ... Я застебнув і переніс сотні концертів і мільйони файлів ... ssh потрібен для ефективного створення архівів.

Але я вірю, що отримані файли можна використовувати з exec безпосередньо з php:

exec('zip -r backup-2013-03-30_0 . -i@backup-2013-03-30_0.txt');

Я не знаю, чи працює. Я не пробував ...

"секрет" полягає в тому, що час виконання архівації не повинен перевищувати час, передбачений для виконання коду PHP.


1

Це робочий приклад створення ZIP-файлів у PHP:

$zip = new ZipArchive();
$zip_name = time().".zip"; // Zip name
$zip->open($zip_name,  ZipArchive::CREATE);
foreach ($files as $file) {
  echo $path = "uploadpdf/".$file;
  if(file_exists($path)){
  $zip->addFromString(basename($path),  file_get_contents($path));---This is main function  
  }
  else{
   echo"file does not exist";
  }
}
$zip->close();

1

Я знайшов цю публікацію в google як другий найкращий результат, спочатку використовував exec :(

У всякому разі, поки це точно не відповідало моїм потребам .. Я вирішив опублікувати відповідь для інших своєю швидкою, але розширеною версією цього.

ОСОБЛИВОСТІ СКРИПТУ

  • Іменування файлів резервного копіювання день у день, PREFIX-YYYY-MM-DD-POSTFIX.EXTENSION
  • Звітність / відсутність файлів
  • Попередній перелік резервних копій
  • Не містить zip / не включає попередні резервні копії;)
  • Працює на windows / linux

У будь-якому випадку, на сценарій .. Хоча це може виглядати як багато .. Пам’ятайте, що тут є надлишок .. Тож сміливо видаляйте розділи звітності за потреби ...

Крім того, це може виглядати безладним, і деякі речі можна легко прибрати ... Тому не коментуйте це, його просто швидкий сценарій з основними коментарями, кинутими .. НЕ ДЛЯ ЖИТТЯ ВИКОРИСТАННЯ .. Але легко прибирати для використання в прямому ефірі !

У цьому прикладі він запускається з каталогу, що знаходиться всередині кореневої папки www / public_html. Отже, для отримання кореня потрібно лише подорожувати однією папкою.

<?php
    // DIRECTORY WE WANT TO BACKUP
    $pathBase = '../';  // Relate Path

    // ZIP FILE NAMING ... This currently is equal to = sitename_www_YYYY_MM_DD_backup.zip 
    $zipPREFIX = "sitename_www";
    $zipDATING = '_' . date('Y_m_d') . '_';
    $zipPOSTFIX = "backup";
    $zipEXTENSION = ".zip";

    // SHOW PHP ERRORS... REMOVE/CHANGE FOR LIVE USE
    ini_set('display_errors',1);
    ini_set('display_startup_errors',1);
    error_reporting(-1);




// ############################################################################################################################
//                                  NO CHANGES NEEDED FROM THIS POINT
// ############################################################################################################################

    // SOME BASE VARIABLES WE MIGHT NEED
    $iBaseLen = strlen($pathBase);
    $iPreLen = strlen($zipPREFIX);
    $iPostLen = strlen($zipPOSTFIX);
    $sFileZip = $pathBase . $zipPREFIX . $zipDATING . $zipPOSTFIX . $zipEXTENSION;
    $oFiles = array();
    $oFiles_Error = array();
    $oFiles_Previous = array();

    // SIMPLE HEADER ;)
    echo '<center><h2>PHP Example: ZipArchive - Mayhem</h2></center>';

    // CHECK IF BACKUP ALREADY DONE
    if (file_exists($sFileZip)) {
        // IF BACKUP EXISTS... SHOW MESSAGE AND THATS IT
        echo "<h3 style='margin-bottom:0px;'>Backup Already Exists</h3><div style='width:800px; border:1px solid #000;'>";
            echo '<b>File Name: </b>',$sFileZip,'<br />';
            echo '<b>File Size: </b>',$sFileZip,'<br />';
        echo "</div>";
        exit; // No point loading our function below ;)
    } else {

        // NO BACKUP FOR TODAY.. SO START IT AND SHOW SCRIPT SETTINGS
        echo "<h3 style='margin-bottom:0px;'>Script Settings</h3><div style='width:800px; border:1px solid #000;'>";
            echo '<b>Backup Directory: </b>',$pathBase,'<br /> ';
            echo '<b>Backup Save File: </b>',$sFileZip,'<br />';
        echo "</div>";

        // CREATE ZIPPER AND LOOP DIRECTORY FOR SUB STUFF
        $oZip = new ZipArchive;
        $oZip->open($sFileZip,  ZipArchive::CREATE | ZipArchive::OVERWRITE);
        $oFilesWrk = new RecursiveIteratorIterator(new RecursiveDirectoryIterator($pathBase),RecursiveIteratorIterator::LEAVES_ONLY);
        foreach ($oFilesWrk as $oKey => $eFileWrk) {
            // VARIOUS NAMING FORMATS OF THE CURRENT FILE / DIRECTORY.. RELATE & ABSOLUTE
            $sFilePath = substr($eFileWrk->getPathname(),$iBaseLen, strlen($eFileWrk->getPathname())- $iBaseLen);
            $sFileReal = $eFileWrk->getRealPath();
            $sFile = $eFileWrk->getBasename();

            // WINDOWS CORRECT SLASHES
            $sMyFP = str_replace('\\', '/', $sFileReal);

            if (file_exists($sMyFP)) {  // CHECK IF THE FILE WE ARE LOOPING EXISTS
                if ($sFile!="."  && $sFile!="..") { // MAKE SURE NOT DIRECTORY / . || ..
                    // CHECK IF FILE HAS BACKUP NAME PREFIX/POSTFIX... If So, Dont Add It,, List It
                    if (substr($sFile,0, $iPreLen)!=$zipPREFIX && substr($sFile,-1, $iPostLen + 4)!= $zipPOSTFIX.$zipEXTENSION) {
                        $oFiles[] = $sMyFP;                     // LIST FILE AS DONE
                        $oZip->addFile($sMyFP, $sFilePath);     // APPEND TO THE ZIP FILE
                    } else {
                        $oFiles_Previous[] = $sMyFP;            // LIST PREVIOUS BACKUP
                    }
                }
            } else {
                $oFiles_Error[] = $sMyFP;                       // LIST FILE THAT DOES NOT EXIST
            }
        }
        $sZipStatus = $oZip->getStatusString();                 // GET ZIP STATUS
        $oZip->close(); // WARNING: Close Required to append files, dont delete any files before this.

        // SHOW BACKUP STATUS / FILE INFO
        echo "<h3 style='margin-bottom:0px;'>Backup Stats</h3><div style='width:800px; height:120px; border:1px solid #000;'>";
            echo "<b>Zipper Status: </b>" . $sZipStatus . "<br />";
            echo "<b>Finished Zip Script: </b>",$sFileZip,"<br />";
            echo "<b>Zip Size: </b>",human_filesize($sFileZip),"<br />";
        echo "</div>";


        // SHOW ANY PREVIOUS BACKUP FILES
        echo "<h3 style='margin-bottom:0px;'>Previous Backups Count(" . count($oFiles_Previous) . ")</h3><div style='overflow:auto; width:800px; height:120px; border:1px solid #000;'>";
        foreach ($oFiles_Previous as $eFile) {
            echo basename($eFile) . ", Size: " . human_filesize($eFile) . "<br />";
        }
        echo "</div>";

        // SHOW ANY FILES THAT DID NOT EXIST??
        if (count($oFiles_Error)>0) {
            echo "<h3 style='margin-bottom:0px;'>Error Files, Count(" . count($oFiles_Error) . ")</h3><div style='overflow:auto; width:800px; height:120px; border:1px solid #000;'>";
            foreach ($oFiles_Error as $eFile) {
                echo $eFile . "<br />";
            }
            echo "</div>";
        }

        // SHOW ANY FILES THAT HAVE BEEN ADDED TO THE ZIP
        echo "<h3 style='margin-bottom:0px;'>Added Files, Count(" . count($oFiles) . ")</h3><div style='overflow:auto; width:800px; height:120px; border:1px solid #000;'>";
        foreach ($oFiles as $eFile) {
            echo $eFile . "<br />";
        }
        echo "</div>";

    }


    // CONVERT FILENAME INTO A FILESIZE AS Bytes/Kilobytes/Megabytes,Giga,Tera,Peta
    function human_filesize($sFile, $decimals = 2) {
        $bytes = filesize($sFile);
        $sz = 'BKMGTP';
        $factor = floor((strlen($bytes) - 1) / 3);
        return sprintf("%.{$decimals}f", $bytes / pow(1024, $factor)) . @$sz[$factor];
    }
?>

ЩО ВОНО РОБИТЬ??

Він просто зафіксує повний вміст змінної $ pathBase і збереже поштовий індекс у тій самій папці. Це просте виявлення попередніх резервних копій та пропускає їх.

CRON BACKUP

Цей скрипт я щойно перевірив на Linux і добре працював із завданням cron, використовуючи абсолютний URL для pathBase.


Я також виключив сценарій видалення, ви можете побачити прийняту відповідь на це
Сердитий 84

Треба любити ці випадкові голоси без коментарів, що пояснюють, чому.
Сердитий 84

1

Використовуйте цю функцію:

function zip($source, $destination)
{
    if (!extension_loaded('zip') || !file_exists($source)) {
        return false;
    }

    $zip = new ZipArchive();
    if (!$zip->open($destination, ZIPARCHIVE::CREATE)) {
        return false;
    }

    $source = str_replace('\\', '/', realpath($source));

    if (is_dir($source) === true) {
        $files = new RecursiveIteratorIterator(new RecursiveDirectoryIterator($source), RecursiveIteratorIterator::SELF_FIRST);

        foreach ($files as $file) {
            $file = str_replace('\\', '/', $file);

            // Ignore "." and ".." folders
            if (in_array(substr($file, strrpos($file, '/')+1), array('.', '..'))) {
                continue;
            }               

            $file = realpath($file);

            if (is_dir($file) === true) {
                $zip->addEmptyDir(str_replace($source . '/', '', $file . '/'));
            } elseif (is_file($file) === true) {
                $zip->addFromString(str_replace($source . '/', '', $file), file_get_contents($file));
            }
        }
    } elseif (is_file($source) === true) {
        $zip->addFromString(basename($source), file_get_contents($source));
    }

    return $zip->close();
}

Приклад використання:

zip('/folder/to/compress/', './compressed.zip');

1

Використовуйте це працює чудово.

$dir = '/Folder/';
$zip = new ZipArchive();
$res = $zip->open(trim($dir, "/") . '.zip', ZipArchive::CREATE | ZipArchive::OVERWRITE);
if ($res === TRUE) {
    foreach (glob($dir . '*') as $file) {
        $zip->addFile($file, basename($file));
    }
    $zip->close();
} else {
    echo 'Failed to create to zip. Error: ' . $res;
}

1

Створіть папку zip в PHP.

Метод створення zip

   public function zip_creation($source, $destination){
    $dir = opendir($source);
    $result = ($dir === false ? false : true);

    if ($result !== false) {

        
        $rootPath = realpath($source);
         
        // Initialize archive object
        $zip = new ZipArchive();
        $zipfilename = $destination.".zip";
        $zip->open($zipfilename, ZipArchive::CREATE | ZipArchive::OVERWRITE );
         
        // Create recursive directory iterator
        /** @var SplFileInfo[] $files */
        $files = new RecursiveIteratorIterator(new RecursiveDirectoryIterator($rootPath), RecursiveIteratorIterator::LEAVES_ONLY);
         
        foreach ($files as $name => $file)
        {
            // Skip directories (they would be added automatically)
            if (!$file->isDir())
            {
                // Get real and relative path for current file
                $filePath = $file->getRealPath();
                $relativePath = substr($filePath, strlen($rootPath) + 1);
         
                // Add current file to archive
                $zip->addFile($filePath, $relativePath);
            }
        }
         
        // Zip archive will be created only after closing object
        $zip->close();
        
        return TRUE;
    } else {
        return FALSE;
    }


}

Викличте zip-метод

$source = $source_directory;
$destination = $destination_directory;
$zipcreation = $this->zip_creation($source, $destination);

0

Я зробив невелике вдосконалення сценарію.

  <?php
    $directory = "./";
    //create zip object
    $zip = new ZipArchive();
    $zip_name = time().".zip";
    $zip->open($zip_name,  ZipArchive::CREATE);
    $files = new RecursiveIteratorIterator(
        new RecursiveDirectoryIterator($directory),
        RecursiveIteratorIterator::LEAVES_ONLY
    );
    foreach ($files as $file) {
        $path = $file->getRealPath();
        //check file permission
        if(fileperms($path)!="16895"){
            $zip->addFromString(basename($path),  file_get_contents($path)) ;
            echo "<span style='color:green;'>{$path} is added to zip file.<br /></span> " ;
        }
        else{
            echo"<span style='color:red;'>{$path} location could not be added to zip<br /></span>";
        }
    }
    $zip->close();
    ?>


0

Це вирішить вашу проблему. Будь ласка, спробуйте.

$zip = new ZipArchive;
$zip->open('testPDFZip.zip', ZipArchive::CREATE);
foreach (glob(APPLICATION_PATH."pages/recruitment/uploads/test_pdf_folder/*") as $file) {
    $new_filename = end(explode("/",$file));
    $zip->addFile($file,"emp/".$new_filename);
}           
$zip->close();

0

Для тих, хто читає цю публікацію та шукає, чому зашпаровувати файли за допомогою addFile замість addFromString, який не копіює файли з їх абсолютним шляхом (просто перетягує файли та нічого іншого), дивіться моє запитання та відповідь тут

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.