18.6.1 Problem
You want to load the entire contents of a file into a
variable. For example, you want to determine if the text in a file matches a
regular expression.
18.6.2 Solution
$fh = fopen('people.txt','r') or die($php_errormsg);
$people = fread($fh,filesize('people.txt'));
if (preg_match('/Names:.*(David|Susannah)/i',$people)) {
print "people.txt matches.";
}
fclose($fh) or die($php_errormsg);
18.6.3 Discussion
$fh = fopen('people.jpg','rb') or die($php_errormsg);
$people = fread($fh,filesize('people.jpg'));
fclose($fh);
There are easier ways to print the entire contents of a file
than by reading it into a string and then printing the string. PHP provides two
functions for this. The first is fpassthru($fh),
which prints everything left on the file handle $fh and then closes it.
The second, readfile($filename), prints the
entire contents of $filename.
You can use readfile( ) to implement a wrapper around
images that shouldn't always be displayed. This program makes sure a requested
image is less than a week old:
$image_directory = '/usr/local/images';
if (preg_match('/^[a-zA-Z0-9]+\.(gif|jpeg)$/',$image,$matches) &&
is_readable($image_directory."/$image") &&
(filemtime($image_directory."/$image") >= (time() - 86400 * 7))) {
header('Content-Type: image/'.$matches[1]);
header('Content-Length: '.filesize($image_directory."/$image"));
readfile($image_directory."/$image");
} else {
error_log("Can't serve image: $image");
}
The directory in which the images are stored,
$image_directory, needs to be outside the web server's document root
for the wrapper to be effective. Otherwise, users can just access the image
files directly. You test the image for three things. First, that the filename
passed in $image is just alphanumeric with an ending of either .gif or .jpeg. You need
to ensure that characters such as .. or / are not in the filename; this
prevents malicious users from retrieving files outside the specified directory.
Second, use is_readable( ) to make sure you can
read the file. Finally, get the file's modification time with filemtime(
) and
make sure that time is after 86400 x 7 seconds ago. There are 86,400 seconds in
a day, so 86400 x 7 is a week.[1] If all of these conditions are met, you're ready to send
the image. First, send two headers to tell the browser the image's MIME type and file size. Then use readfile( ) to
send the entire contents of the file to the user.
[1] When switching between standard time and daylight saving time, there are not 86,400 seconds in a day. See Section 3.11 for details.