Friday, 10 August 2018

Save Browser Output To A File With PHP Output Buffering Functions

The PHP output buffering functions provide a handy way of intercepting the contents of the buffer before it is sent to the browser. The output is whatever is sent to the browser whenever you print something off. PHP allows you to capture this output in a buffer before it is sent to the browser.
Output buffering is controlled by two mechanisms. The first is through the php.ini directive output_buffering, which is usually set to off. It can be turned on by setting this to either on, or the number of bytes that the buffer can take up. When this byte allocation is full the output is sent to the browser.
The second mechanism is through the use of the ob_start() function, which turns on output buffering for the script it is being run on. If output buffering has been set in the php.ini file then this isn't needed, but if it has been set to 'off' then you will get some errors if you try to do anything with the buffer. To start output buffering just call the ob_start() function.
ob_start();
Whatever you try to print off now will not be output to the browser until the end of the script. To explicitly push the output to the browser you can call the ob_flush() function.
  1. echo 'test';
You can turn off output buffering at any time by calling either ob_end_clean() or ob_end_flush(). ob_end_clean() will turn off the top most buffer (you can have more than one) and discard the contents. ob_end_flush() will output the contents of the top most buffer then turn it off. Here is an easy way to use ob_end_flush() to flush and turn off all available output buffers and output their contents.
while (@ob_end_flush());
You can get the contents of the output buffer at any time (as long as you haven't turned it off) by calling the ob_get_contents() function. This returns the current buffer contents. Here is an example of it in action.
  1. <?php ob_start();
  2. echo 'test ';
  3. ??>
  4. some more text
  5. <?php $out = ob_get_contents();
  6. // $out now contains 'test some more text'.
  7. ??>
The ob_flush() function will empty the current output buffer and send this to the browser. So it is important to call this function after you have got the contents of the buffer. If you call it before you try to get the buffer contents there will be nothing there.
Using the ob_get_contents() function is how you can output text to both a file and the browser. The following example will write some text to the browser and also store this text in a variable that is written to a file at the end of the script. The file is given the current time stamp as a name. Make sure you have sufficient rights to write into this directory using chmod.
  1. $out = '';
  2. echo 'test';
  3. $out .= ob_get_contents();
  4. echo ' some more text to add to the buffer';
  5. $out .= ob_get_contents();
  6. // check that something was actually written to the buffer
  7. if (strlen($out) > 0) {
  8. $file = 'debug/' . time() . '.html';
  9. touch($file);
  10. $fh = fopen($file, 'w');
  11. fwrite($fh, $out);
  12. fclose($fh);
  13. }
The ob_start() function can be called without any parameters, but you can supply the name of a function that will be used to pass the contents of the buffer through before it is sent to the browser. Using this mechanism is slightly easier as it can be easily removed from any script. The call back function must have a single string parameter, which is the contents of the buffer, and must return a string, preferably the contents of the buffer.
  1. function bufferCallBack($buffer)
  2. {
  3. // check that something was actually written to the buffer
  4. if (strlen($buffer) > 0) {
  5. $file = 'debug/' . time() . '.html';
  6. touch($file);
  7. $fh = fopen($file, 'w');
  8. fwrite($fh, $buffer);
  9. fclose($fh);
  10. }
  11. return $buffer;
  12. }
  13.  
  14. ob_start('bufferCallBack');
  15. echo 'test';
  16. echo ' some more text to add to the buffer';
Be warned that the use of output buffering can lead to the apparent slowdown of your website due to everything being generated server side before being sent to the client.
Update: Added some advice about having correct permissions and corrected a variable name in the second example. Thanks to William for his email on this :)

0 comments:

Post a Comment