How to Zip and Unzip Files in PHP
Compressing files when transferring them over the internet has a lot of advantages. In most cases, the combined total size of all the files in the compressed format comes down by a nice margin. This means that you will save some of your bandwidth, and users will also get faster download speeds. Once the users have downloaded a file, they can decompress it whenever they want. In short, compression can make serving files over the internet a lot easier for you as well as your visitors.
One factor that can discourage you from compressing files or make the process very tiresome is the fact that you might be doing it manually. Luckily, PHP comes with a lot of extensions that deal specifically with file compression and extraction. You can use the functions available in these extensions to automatically compress files in PHP.
This tutorial will teach you how to zip and unzip (compress and extract) files to and from a zip archive in PHP. You will also learn how to delete or rename files in an archive without extracting them first.
Compressing Files in PHP
The PHP ZipArchive
class has a lot of properties and methods which can help you compress and decompress all your files.
Compress Individual Files
You can add files to your zip archive one at a time or add the whole directory at once. In either case, the first step is creating a new ZipArchive
instance and then calling the open($filename, [$flags])
method. This method will open a new zip archive for reading, writing, or other modifications. There are four valid values for the optional $flag
parameter which determine how to handle different situations.
ZipArchive::OVERWRITE
—This flag will overwrite the contents in the specified archive if it already exists.ZipArchive::CREATE
—This flag will create a new archive if it does not already exist.ZipArchive::EXCL
—This flag will result in an error if the archive already exists.ZipArchive::CHECKCONS
—This flag will tell PHP to perform additional consistency checks on the archive and give an error if they fail.
You can check the documentation of this method to learn about different error codes returned in case of failures to open the file. If the zip file was opened or created successfully, the method will return true
.
Once you have opened the archive successfully, you can use the addFile($filename, $localname, $start, $length)
method to add any file from a given path to your archive. The $filename
parameter is the path of a file that you want to add to the archive. The $localname
parameter is used to assign a name to the file to store it inside the archive. You can call addFile()
every time you want to add a new file to your archive.
After adding all the necessary files to the archive, you can simply call the close()
method to close it and save the changes.
Let's say you have a website which allows users to download font files for different fonts along with the licensing information to use them. Files like these will be perfect examples of automated archiving using PHP. The following code shows you how to do exactly that.
<?php $zip = new ZipArchive(); $zip->open('compressed/font_files.zip', ZipArchive::CREATE); $zip->addFile('fonts/Monoton/Monoton-Regular.ttf', 'Monoton-Regular.ttf'); $zip->addFile('fonts/Monoton/OFL.txt', 'license.txt'); $zip->close(); ?>
We begin by creating a ZipArchive
instance and then using the open()
method to create our archive. The addFile()
method adds our actual .ttf font file and the .txt license file to the archive.
You should note that the original files were inside the fonts/Monoton directory. However, the PHP code places it directly inside the root of our archive. You can change the directory structure as well as the names of files going in the archive.
Compressing Multiple Files From a Directory
Adding individual files to your archive can get tiring after a while. For example, you might want to create an archive of all .pdf or .png files in a directory. The addGlob($pattern, $flags, $options)
method will prove very helpful in this case. The only disadvantage of this method is that you lose control over the location of individual files in the archive. However, you can still influence the directory structure inside the archive using the $options
parameter. The options are passed in the form of an associative array.
add_path
—The value you assign toadd_path
is prefixed to the local path of the file within the archive.remove_path
—The value you assign toremove_path
is used to remove a matching prefix from the path of different files which are added to the archive.remove_all_path
—Setting the value ofremove_all_path
totrue
will remove everything from the path of the file besides its name. In this case, the files are added to the root of the archive.
It's important to remember that removal of a path is done before prefixing the value specified in add_path
.
The following code snippet will make the use of addGlob()
and all these options clearer.
$zip = new ZipArchive(); $zip->open('compressed/user_archive.zip', ZipArchive::CREATE); $options = array('add_path' => 'light_wallpapers/', 'remove_all_path' => TRUE); $zip->addGlob('lights/*.jpg', 0, $options); $options = array('add_path' => 'font_files/', 'remove_all_path' => TRUE); $zip->addGlob('documents/*.ttf', 0, $options); $options = array('add_path' => 'pdf_books/', 'remove_all_path' => TRUE); $zip->addGlob('documents/*.pdf', 0, $options); $options = array('add_path' => 'images/', 'remove_all_path' => TRUE); $zip->addGlob('documents/*.{jpg, png}', GLOB_BRACE, $options); $zip->close();
As usual, we begin by creating a ZipArchive
instance and then use the open()
method to create our archive. We also specify different values for the add_path
key in the $options
array each time before calling the addGlob()
method. This way, we can deal with one specific set of files at a time and provide archiving options accordingly.
In the first case, we iterate over all .jpg files in the lights directory and place them in the light_wallpapers directory in the archive. Similarly, we iterate over all the .ttf files in the documents directory and then put them inside a folder called font_files in our archive. Finally, we iterate over all the .jpg and .png files in our documents at once and put them all together in the images directory.
As you can see, the values in the $options
parameter are useful in organizing the content inside the archive.
Extracting Content From an Archive
The ZipArchive
class has a method called extractTo($destination, $entries)
to extract the contents of an archive. You can use it to either extract everything inside the archive or just some specific files. The $entries
parameter can be used to specify a single file name which is to be extracted, or you can use it to pass an array of files.
One important point to remember is that you need to specify the proper path of the file inside the archive in order to extract it. For example, we archived a font file called AlegreyaSans-Light.ttf in the previous section. The file was stored within the archive in a directory called font_files. This means that the path you need to specify in the $entries
parameter would be font_files/AlegreyaSans-Light.ttf and not simply AlegreyaSans-Light.ttf.
The directory and file structure will be preserved during the extraction process, and files will be extracted in their respective directories.
<?php $zip = new ZipArchive(); $zip->open('compressed/user_archive.zip', ZipArchive::CREATE); $zip->extractTo('uncompressed/', 'font_files/AlegreyaSans-Light.ttf'); $zip->close(); ?>
If you omit the second parameter, the method will extract all files in the archive.
Get More Control Over the Archives
The ZipArchive
class also has a lot of other methods and properties to help you get more information about the archive before extracting all its contents.
You can count the number of files in an archive using the count()
method. Another option is to use the numFiles
property. They can be used to iterate over all the files in the archive and only extract the ones you need—or you can do something else with them, like removing them from the archive.
In the following example, we are deleting all files in the archive which contain the word Italic. Similar code could be used to delete all files which don't contain a specific word. You could also iterate over these files and replace a particular word with something else.
<?php $zip = new ZipArchive(); $zip->open('compressed/user_archive.zip', ZipArchive::CREATE); $file_count = $zip->count(); for($i = 0; $i < $file_count; $i++) { $file_name = $zip->getNameIndex($i); if(stripos($file_name, 'Italic') !== false) { $zip->deleteName($file_name); } } $zip->close(); ?>
In the above code, we are using deleteName()
to delete an individual file. However, you can also use it to delete an entire directory.
A similar function renameName($oldname, $newname)
can be used to change the name of any files in the archive. You will get an error if a file titled $newname
already exists.
Final Thoughts
We have covered a bunch of very useful methods of the ZipArchive
class which will make automated compression and extraction of files in PHP a breeze. You should now be able to compress individual files or a group of them at once, based on your own criteria. Similarly, you should be able to extract any particular file from the archive without affecting other content.
With the help of count()
and numFiles
, you will get more control over the individual files, and renaming or deleting them would be super easy. You should go through the documentation at least once to read about more such functions.
from Envato Tuts+ Tutorials
Comments
Post a Comment