sintonia/legacy/application/common/FileIO.php

89 lines
3.3 KiB
PHP

<?php
/**
* Class Application_Common_FileIO contains helper functions for reading and writing files, and sending them over HTTP.
*/
class Application_Common_FileIO
{
/**
* Reads the requested portion of a file and sends its contents to the client with the appropriate headers.
*
* This HTTP_RANGE compatible read file function is necessary for allowing streaming media to be skipped around in.
*
* @param string $filePath - the full filepath or URL pointing to the location of the file
* @param string $mimeType - the file's mime type. Defaults to 'audio/mp3'
* @param int $size - the file size, in bytes
*
* @see https://groups.google.com/d/msg/jplayer/nSM2UmnSKKA/Hu76jDZS4xcJ
* @see https://php.net/manual/en/function.readfile.php#86244
*/
public static function smartReadFile($filePath, $size, $mimeType)
{
$fm = @fopen($filePath, 'rb');
if (!$fm) {
throw new LibreTimeFileNotFoundException($filePath);
}
// Note that $size is allowed to be zero. If that's the case, it means we don't
// know the filesize, and we need to figure one out so modern browsers don't get
// confused. This should only affect files imported by legacy upstream since
// media monitor did not always set the proper size in the database but analyzer
// seems to always have a value for this.
if ($size === 0) {
$fstats = fstat($fm);
$size = $fstats['size'];
}
if ($size <= 0) {
throw new Exception("Invalid file size returned for file at {$filePath}");
}
$begin = 0;
$end = $size - 1;
ob_start(); // Must start a buffer here for these header() functions
if (isset($_SERVER['HTTP_RANGE'])) {
if (preg_match('/bytes=\h*(\d+)-(\d*)[\D.*]?/i', $_SERVER['HTTP_RANGE'], $matches)) {
$begin = intval($matches[1]);
if (!empty($matches[2])) {
$end = intval($matches[2]);
}
}
}
if (isset($_SERVER['HTTP_RANGE'])) {
header('HTTP/1.1 206 Partial Content');
} else {
header('HTTP/1.1 200 OK');
}
header("Content-Type: {$mimeType}");
header('Content-Transfer-Encoding: binary');
header('Cache-Control: public, must-revalidate, max-age=0');
header('Pragma: no-cache');
header('Accept-Ranges: bytes');
header('Content-Length:' . (($end - $begin) + 1));
if (isset($_SERVER['HTTP_RANGE'])) {
header("Content-Range: bytes {$begin}-{$end}/{$size}");
}
// We can have multiple levels of output buffering. Need to
// keep looping until all have been disabled!!!
// https://www.php.net/manual/en/function.ob-end-flush.php
while (ob_get_level() > 0) {
ob_end_flush();
}
// These two lines were removed from Airtime 2.5.x at some point after Libretime forked from Airtime.
// These lines allow seek to work for files.
// Issue #349
$cur = $begin;
fseek($fm, $begin, 0);
while (!feof($fm) && (connection_status() == 0) && ($cur <= $end)) {
echo fread($fm, 1024 * 8);
}
fclose($fm);
}
}