Recipe 18.6 Reading a File into a String
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
Use filesize( ) to get the size of the file, and then tell
fread( ) to read that many bytes:
$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
To read a binary file (e.g., an image) on
Windows, a b must be appended to the file mode:
$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. 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.
18.6.4 See Also
Documentation on filesize( ) at http://www.php.net/filesize, fread(
) at http://www.php.net/fread, fpassthru(
) at http://www.php.net/fpassthru, and
readfile( ) at http://www.php.net/readfile.
|