mirror of
				https://github.com/lubuntu-team/lubuntu.me.git
				synced 2025-10-25 21:54:03 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			267 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			267 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| 
 | |
| // https://stackoverflow.com/questions/157318/resumable-downloads-when-using-php-to-send-the-file
 | |
| // User: DaveRandom
 | |
| 
 | |
| if (!class_exists('UpdraftPlus_NonExistentFileException')):
 | |
| class UpdraftPlus_NonExistentFileException extends RuntimeException {}
 | |
| endif;
 | |
| 
 | |
| if (!class_exists('UpdraftPlus_UnreadableFileException')):
 | |
| class UpdraftPlus_UnreadableFileException extends RuntimeException {}
 | |
| endif;
 | |
| 
 | |
| if (!class_exists('UpdraftPlus_UnsatisfiableRangeException')):
 | |
| class UpdraftPlus_UnsatisfiableRangeException extends RuntimeException {}
 | |
| endif;
 | |
| 
 | |
| if (!class_exists('UpdraftPlus_InvalidRangeHeaderException')):
 | |
| class UpdraftPlus_InvalidRangeHeaderException extends RuntimeException {}
 | |
| endif;
 | |
| 
 | |
| class UpdraftPlus_RangeHeader
 | |
| {
 | |
| 	/**
 | |
| 	 * The first byte in the file to send (0-indexed), a null value indicates the last
 | |
| 	 * $end bytes
 | |
| 	 *
 | |
| 	 * @var int|null
 | |
| 	 */
 | |
| 	private $firstByte;
 | |
| 
 | |
| 	/**
 | |
| 	 * The last byte in the file to send (0-indexed), a null value indicates $start to
 | |
| 	 * EOF
 | |
| 	 *
 | |
| 	 * @var int|null
 | |
| 	 */
 | |
| 	private $lastByte;
 | |
| 
 | |
| 	/**
 | |
| 	 * Create a new instance from a Range header string
 | |
| 	 *
 | |
| 	 * @param string $header
 | |
| 	 * @return UpdraftPlus_RangeHeader
 | |
| 	 */
 | |
| 	public static function createFromHeaderString($header)
 | |
| 	{
 | |
| 		if ($header === null) {
 | |
| 			return null;
 | |
| 		}
 | |
| 		if (!preg_match('/^\s*([A-Za-z]+)\s*=\s*(\d*)\s*-\s*(\d*)\s*(?:,|$)/', $header, $info)) {
 | |
| 			throw new UpdraftPlus_InvalidRangeHeaderException('Invalid header format');
 | |
| 		} else if (strtolower($info[1]) !== 'bytes') {
 | |
| 			throw new UpdraftPlus_InvalidRangeHeaderException('Unknown range unit: ' . $info[1]);
 | |
| 		}
 | |
| 
 | |
| 		return new self(
 | |
| 			$info[2] === '' ? null : $info[2],
 | |
| 			$info[3] === '' ? null : $info[3]
 | |
| 		);
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * @param int|null $firstByte
 | |
| 	 * @param int|null $lastByte
 | |
| 	 * @throws UpdraftPlus_InvalidRangeHeaderException
 | |
| 	 */
 | |
| 	public function __construct($firstByte, $lastByte)
 | |
| 	{
 | |
| 		$this->firstByte = $firstByte === null ? $firstByte : (int)$firstByte;
 | |
| 		$this->lastByte = $lastByte === null ? $lastByte : (int)$lastByte;
 | |
| 
 | |
| 		if ($this->firstByte === null && $this->lastByte === null) {
 | |
| 			throw new UpdraftPlus_InvalidRangeHeaderException(
 | |
| 				'Both start and end position specifiers empty'
 | |
| 			);
 | |
| 		} else if ($this->firstByte < 0 || $this->lastByte < 0) {
 | |
| 			throw new UpdraftPlus_InvalidRangeHeaderException(
 | |
| 				'Position specifiers cannot be negative'
 | |
| 			);
 | |
| 		} else if ($this->lastByte !== null && $this->lastByte < $this->firstByte) {
 | |
| 			throw new UpdraftPlus_InvalidRangeHeaderException(
 | |
| 				'Last byte cannot be less than first byte'
 | |
| 			);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the start position when this range is applied to a file of the specified size
 | |
| 	 *
 | |
| 	 * @param int $fileSize
 | |
| 	 * @return int
 | |
| 	 * @throws UpdraftPlus_UnsatisfiableRangeException
 | |
| 	 */
 | |
| 	public function getStartPosition($fileSize)
 | |
| 	{
 | |
| 		$size = (int)$fileSize;
 | |
| 
 | |
| 		if ($this->firstByte === null) {
 | |
| 			return ($size - 1) - $this->lastByte;
 | |
| 		}
 | |
| 
 | |
| 		if ($size <= $this->firstByte) {
 | |
| 			throw new UpdraftPlus_UnsatisfiableRangeException(
 | |
| 				'Start position is after the end of the file'
 | |
| 			);
 | |
| 		}
 | |
| 
 | |
| 		return $this->firstByte;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the end position when this range is applied to a file of the specified size
 | |
| 	 *
 | |
| 	 * @param int $fileSize
 | |
| 	 * @return int
 | |
| 	 * @throws UpdraftPlus_UnsatisfiableRangeException
 | |
| 	 */
 | |
| 	public function getEndPosition($fileSize)
 | |
| 	{
 | |
| 		$size = (int)$fileSize;
 | |
| 
 | |
| 		if ($this->lastByte === null) {
 | |
| 			return $size - 1;
 | |
| 		}
 | |
| 
 | |
| 		if ($size <= $this->lastByte) {
 | |
| 			throw new UpdraftPlus_UnsatisfiableRangeException(
 | |
| 				'End position is after the end of the file'
 | |
| 			);
 | |
| 		}
 | |
| 
 | |
| 		return $this->lastByte;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get the length when this range is applied to a file of the specified size
 | |
| 	 *
 | |
| 	 * @param int $fileSize
 | |
| 	 * @return int
 | |
| 	 * @throws UpdraftPlus_UnsatisfiableRangeException
 | |
| 	 */
 | |
| 	public function getLength($fileSize)
 | |
| 	{
 | |
| 		$size = (int)$fileSize;
 | |
| 
 | |
| 		return $this->getEndPosition($size) - $this->getStartPosition($size) + 1;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get a Content-Range header corresponding to this Range and the specified file
 | |
| 	 * size
 | |
| 	 *
 | |
| 	 * @param int $fileSize
 | |
| 	 * @return string
 | |
| 	 */
 | |
| 	public function getContentRangeHeader($fileSize)
 | |
| 	{
 | |
| 		return 'bytes ' . $this->getStartPosition($fileSize) . '-'
 | |
| 			 . $this->getEndPosition($fileSize) . '/' . $fileSize;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| class UpdraftPlus_PartialFileServlet
 | |
| {
 | |
| 	/**
 | |
| 	 * The range header on which the data transmission will be based
 | |
| 	 *
 | |
| 	 * @var UpdraftPlus_RangeHeader|null
 | |
| 	 */
 | |
| 	private $range;
 | |
| 
 | |
| 	/**
 | |
| 	 * @param UpdraftPlus_RangeHeader $range Range header on which the transmission will be based
 | |
| 	 */
 | |
| 	public function __construct($range = null)
 | |
| 	{
 | |
| 		$this->range = $range;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Send part of the data in a seekable stream resource to the output buffer
 | |
| 	 *
 | |
| 	 * @param resource $fp Stream resource to read data from
 | |
| 	 * @param int $start Position in the stream to start reading
 | |
| 	 * @param int $length Number of bytes to read
 | |
| 	 * @param int $chunkSize Maximum bytes to read from the file in a single operation
 | |
| 	 */
 | |
| 	private function sendDataRange($fp, $start, $length, $chunkSize = 2097152)
 | |
| 	{
 | |
| 		if ($start > 0) {
 | |
| 			fseek($fp, $start, SEEK_SET);
 | |
| 		}
 | |
| 
 | |
| 		while ($length) {
 | |
| 			$read = ($length > $chunkSize) ? $chunkSize : $length;
 | |
| 			$length -= $read;
 | |
| 			echo fread($fp, $read);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Send the headers that are included regardless of whether a range was requested
 | |
| 	 *
 | |
| 	 * @param string $fileName
 | |
| 	 * @param int $contentLength
 | |
| 	 * @param string $contentType
 | |
| 	 */
 | |
| 	private function sendDownloadHeaders($fileName, $contentLength, $contentType)
 | |
| 	{
 | |
| 		header('Content-Type: ' . $contentType);
 | |
| 		header('Content-Length: ' . $contentLength);
 | |
| 		header('Content-Disposition: attachment; filename="' . $fileName . '"');
 | |
| 		header('Accept-Ranges: bytes');
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Send data from a file based on the current Range header
 | |
| 	 *
 | |
| 	 * @param string $path Local file system path to serve
 | |
| 	 * @param string $contentType MIME type of the data stream
 | |
| 	 */
 | |
| 	public function sendFile($path, $contentType = 'application/octet-stream')
 | |
| 	{
 | |
| 		// Make sure the file exists and is a file, otherwise we are wasting our time
 | |
| 		$localPath = realpath($path);
 | |
| 		if ($localPath === false || !is_file($localPath)) {
 | |
| 			throw new UpdraftPlus_NonExistentFileException(
 | |
| 				$path . ' does not exist or is not a file'
 | |
| 			);
 | |
| 		}
 | |
| 
 | |
| 		// Make sure we can open the file for reading
 | |
| 		if (!$fp = fopen($localPath, 'r')) {
 | |
| 			throw new UpdraftPlus_UnreadableFileException(
 | |
| 				'Failed to open ' . $localPath . ' for reading'
 | |
| 			);
 | |
| 		}
 | |
| 
 | |
| 		$fileSize = filesize($localPath);
 | |
| 
 | |
| 		if ($this->range == null) {
 | |
| 			// No range requested, just send the whole file
 | |
| 			header('HTTP/1.1 200 OK');
 | |
| 			$this->sendDownloadHeaders(basename($localPath), $fileSize, $contentType);
 | |
| 
 | |
| 			fpassthru($fp);
 | |
| 		} else {
 | |
| 			// Send the request range
 | |
| 			header('HTTP/1.1 206 Partial Content');
 | |
| 			header('Content-Range: ' . $this->range->getContentRangeHeader($fileSize));
 | |
| 			$this->sendDownloadHeaders(
 | |
| 				basename($localPath),
 | |
| 				$this->range->getLength($fileSize),
 | |
| 				$contentType
 | |
| 			);
 | |
| 
 | |
| 			$this->sendDataRange(
 | |
| 				$fp,
 | |
| 				$this->range->getStartPosition($fileSize),
 | |
| 				$this->range->getLength($fileSize)
 | |
| 			);
 | |
| 		}
 | |
| 
 | |
| 		fclose($fp);
 | |
| 	}
 | |
| } |