alltube/classes/Video.php

573 lines
16 KiB
PHP
Raw Normal View History

2014-03-13 19:07:56 +00:00
<?php
/**
2016-09-07 22:28:28 +00:00
* VideoDownload class.
2016-08-01 11:29:13 +00:00
*/
2016-12-05 12:12:27 +00:00
2015-10-29 19:43:43 +00:00
namespace Alltube;
2016-03-29 23:49:08 +00:00
use Exception;
use GuzzleHttp\Client;
use GuzzleHttp\Psr7\Response;
use stdClass;
use Symfony\Component\Process\Process;
2016-03-31 22:42:28 +00:00
2014-03-13 19:07:56 +00:00
/**
2016-09-07 22:28:28 +00:00
* Extract info about videos.
*
2019-04-21 17:10:37 +00:00
* Due to the way youtube-dl behaves, this class can also contain information about a playlist.
*
* @property-read string $title Title
* @property-read string $protocol Network protocol (HTTP, RTMP, etc.)
* @property-read string $url File URL
* @property-read string $ext File extension
* @property-read string $extractor_key youtube-dl extractor class used
* @property-read array $entries List of videos (if the object contains information about a playlist)
* @property-read array $rtmp_conn
* @property-read string|null $_type Object type (usually "playlist" or null)
2016-08-01 11:29:13 +00:00
*/
class Video
2014-03-13 19:07:56 +00:00
{
2016-10-14 00:40:33 +00:00
/**
2016-10-14 00:40:57 +00:00
* Config instance.
*
2016-10-14 00:40:33 +00:00
* @var Config
*/
2016-10-10 19:32:07 +00:00
private $config;
2016-10-14 00:40:33 +00:00
/**
2019-04-21 16:35:24 +00:00
* URL of the page containing the video.
*
* @var string
*/
private $webpageUrl;
/**
2019-04-21 16:35:24 +00:00
* Requested video format.
*
* @var string
*/
private $requestedFormat;
/**
2019-04-21 16:35:24 +00:00
* Password.
*
* @var string|null
*/
private $password;
2019-04-21 17:10:37 +00:00
/**
* JSON object returned by youtube-dl.
*
* @var stdClass
*/
private $json;
2016-08-01 11:29:13 +00:00
/**
2016-09-07 22:28:28 +00:00
* VideoDownload constructor.
2017-12-19 14:20:52 +00:00
*
* @param string $webpageUrl URL of the page containing the video
* @param string $requestedFormat Requested video format
* @param string $password Password
2016-08-01 11:29:13 +00:00
*/
public function __construct($webpageUrl, $requestedFormat = 'best', $password = null)
2016-04-08 17:06:41 +00:00
{
$this->webpageUrl = $webpageUrl;
$this->requestedFormat = $requestedFormat;
$this->password = $password;
$this->config = Config::getInstance();
}
/**
2017-12-23 13:37:29 +00:00
* Return a youtube-dl process with the specified arguments.
*
* @param string[] $arguments Arguments
*
* @return Process
*/
private static function getProcess(array $arguments)
{
$config = Config::getInstance();
2019-04-21 16:35:24 +00:00
return new Process(
array_merge(
[$config->python, $config->youtubedl],
$config->params,
$arguments
)
);
2016-04-08 17:06:41 +00:00
}
2014-03-13 19:07:56 +00:00
/**
2016-09-07 22:28:28 +00:00
* List all extractors.
*
2016-08-01 11:29:13 +00:00
* @return string[] Extractors
2014-03-13 19:07:56 +00:00
* */
public static function getExtractors()
2014-03-13 19:07:56 +00:00
{
return explode("\n", trim(self::callYoutubedl(['--list-extractors'])));
2014-03-13 19:07:56 +00:00
}
2016-10-14 00:40:33 +00:00
/**
* Call youtube-dl.
2016-10-14 00:40:57 +00:00
*
* @param array $arguments Arguments
2016-10-14 00:40:57 +00:00
*
2017-12-19 14:20:52 +00:00
* @throws PasswordException If the video is protected by a password and no password was specified
* @throws Exception If the password is wrong
* @throws Exception If youtube-dl returns an error
2017-12-19 14:20:52 +00:00
*
* @return string Result
2016-10-14 00:40:33 +00:00
*/
private static function callYoutubedl(array $arguments)
2014-03-13 19:07:56 +00:00
{
$config = Config::getInstance();
$process = self::getProcess($arguments);
//This is needed by the openload extractor because it runs PhantomJS
$process->setEnv(['PATH'=>$config->phantomjsDir]);
$process->inheritEnvironmentVariables();
2016-03-31 22:42:28 +00:00
$process->run();
if (!$process->isSuccessful()) {
$errorOutput = trim($process->getErrorOutput());
$exitCode = $process->getExitCode();
if ($errorOutput == 'ERROR: This video is protected by a password, use the --video-password option') {
throw new PasswordException($errorOutput, $exitCode);
} elseif (substr($errorOutput, 0, 21) == 'ERROR: Wrong password') {
throw new Exception(_('Wrong password'), $exitCode);
} else {
throw new Exception($errorOutput, $exitCode);
}
2014-03-18 14:08:16 +00:00
} else {
return trim($process->getOutput());
2014-03-18 14:08:16 +00:00
}
2014-03-13 19:07:56 +00:00
}
/**
* Get a property from youtube-dl.
*
* @param string $prop Property
*
* @return string
*/
private function getProp($prop = 'dump-json')
{
$arguments = ['--'.$prop];
if (isset($this->webpageUrl)) {
$arguments[] = $this->webpageUrl;
}
if (isset($this->requestedFormat)) {
$arguments[] = '-f';
$arguments[] = $this->requestedFormat;
}
if (isset($this->password)) {
$arguments[] = '--video-password';
$arguments[] = $this->password;
}
return $this::callYoutubedl($arguments);
}
2016-10-13 14:40:19 +00:00
/**
* Get all information about a video.
*
* @return stdClass Decoded JSON
2016-10-13 14:40:19 +00:00
* */
public function getJson()
{
if (!isset($this->json)) {
$this->json = json_decode($this->getProp('dump-single-json'));
}
return $this->json;
}
/**
* Magic method to get a property from the JSON object returned by youtube-dl.
*
* @param string $name Property
*
* @return mixed
*/
public function __get($name)
{
if (isset($this->$name)) {
return $this->getJson()->$name;
}
}
/**
* Magic method to check if the JSON object returned by youtube-dl has a property.
*
* @param string $name Property
*
2019-04-21 16:35:24 +00:00
* @return bool
*/
public function __isset($name)
2016-10-13 14:40:19 +00:00
{
return isset($this->getJson()->$name);
2016-10-13 14:40:19 +00:00
}
2014-03-13 19:07:56 +00:00
/**
2016-09-07 22:28:28 +00:00
* Get URL of video from URL of page.
*
2017-04-24 22:40:24 +00:00
* It generally returns only one URL.
* But it can return two URLs when multiple formats are specified
* (eg. bestvideo+bestaudio).
*
2017-04-25 12:55:21 +00:00
* @return string[] URLs of video
2014-03-13 19:07:56 +00:00
* */
public function getUrl()
2014-03-13 19:07:56 +00:00
{
$urls = explode("\n", $this->getProp('get-url'));
if (empty($urls[0])) {
throw new EmptyUrlException(_('youtube-dl returned an empty URL.'));
}
return $urls;
2014-03-13 19:07:56 +00:00
}
2016-08-01 11:29:13 +00:00
/**
2016-09-07 22:28:28 +00:00
* Get filename of video file from URL of page.
2016-08-01 11:29:13 +00:00
*
* @return string Filename of extracted video
* */
public function getFilename()
{
return trim($this->getProp('get-filename'));
}
2016-08-01 11:29:13 +00:00
/**
2017-04-24 22:41:49 +00:00
* Get filename of video with the specified extension.
2016-08-01 11:29:13 +00:00
*
2017-04-24 22:40:24 +00:00
* @param string $extension New file extension
2016-08-01 11:29:13 +00:00
*
2017-04-24 22:40:24 +00:00
* @return string Filename of extracted video with specified extension
*/
public function getFileNameWithExtension($extension)
{
return html_entity_decode(
pathinfo(
$this->getFilename(),
PATHINFO_FILENAME
2017-04-24 22:40:24 +00:00
).'.'.$extension,
ENT_COMPAT,
'ISO-8859-1'
);
}
2016-10-15 14:18:04 +00:00
/**
* Return arguments used to run rtmp for a specific video.
2016-10-15 14:18:04 +00:00
*
2017-12-23 14:14:43 +00:00
* @return array Arguments
2016-10-15 14:18:04 +00:00
*/
private function getRtmpArguments()
2016-10-15 14:18:04 +00:00
{
2017-12-23 15:04:55 +00:00
$arguments = [];
if ($this->protocol == 'rtmp') {
2018-07-03 18:09:45 +00:00
foreach ([
'url' => '-rtmp_tcurl',
'webpage_url' => '-rtmp_pageurl',
'player_url' => '-rtmp_swfverify',
'flash_version' => '-rtmp_flashver',
'play_path' => '-rtmp_playpath',
'app' => '-rtmp_app',
] as $property => $option) {
if (isset($this->{$property})) {
2018-07-03 18:09:45 +00:00
$arguments[] = $option;
$arguments[] = $this->{$property};
2018-07-03 18:09:45 +00:00
}
2016-10-15 14:18:04 +00:00
}
2016-10-15 14:20:54 +00:00
if (isset($this->rtmp_conn)) {
foreach ($this->rtmp_conn as $conn) {
2018-07-03 18:09:45 +00:00
$arguments[] = '-rtmp_conn';
$arguments[] = $conn;
}
2016-10-14 17:01:51 +00:00
}
}
2016-10-14 17:02:14 +00:00
2017-12-23 14:14:43 +00:00
return $arguments;
2016-10-14 17:01:51 +00:00
}
/**
2017-05-13 22:54:47 +00:00
* Check if a command runs successfully.
*
* @param array $command Command and arguments
*
* @return bool False if the command returns an error, true otherwise
*/
public static function checkCommand(array $command)
{
$process = new Process($command);
$process->run();
return $process->isSuccessful();
}
2016-10-14 17:01:51 +00:00
/**
* Get a process that runs avconv in order to convert a video.
2016-10-14 17:02:14 +00:00
*
2019-04-21 16:35:24 +00:00
* @param int $audioBitrate Audio bitrate of the converted file
* @param string $filetype Filetype of the converted file
* @param bool $audioOnly True to return an audio-only file
* @param string $from Start the conversion at this time
* @param string $to End the conversion at this time
2016-10-14 17:02:14 +00:00
*
* @throws Exception If avconv/ffmpeg is missing
2017-12-19 14:20:52 +00:00
*
2017-12-24 00:12:47 +00:00
* @return Process Process
2016-10-14 17:01:51 +00:00
*/
2018-07-03 17:47:35 +00:00
private function getAvconvProcess(
$audioBitrate,
$filetype = 'mp3',
$audioOnly = true,
$from = null,
$to = null
) {
if (!$this->checkCommand([$this->config->avconv, '-version'])) {
throw new Exception(_('Can\'t find avconv or ffmpeg at ').$this->config->avconv.'.');
2016-10-14 17:01:51 +00:00
}
2016-10-14 17:02:14 +00:00
2018-07-03 17:47:35 +00:00
$durationRegex = '/(\d+:)?(\d+:)?(\d+)/';
2018-07-03 18:09:45 +00:00
$afterArguments = [];
2017-12-23 14:14:43 +00:00
if ($audioOnly) {
2018-07-03 18:09:45 +00:00
$afterArguments[] = '-vn';
2018-07-03 17:47:35 +00:00
}
2018-07-03 18:09:45 +00:00
if (!empty($from)) {
2018-07-03 17:47:35 +00:00
if (!preg_match($durationRegex, $from)) {
throw new Exception(_('Invalid start time: ').$from.'.');
}
$afterArguments[] = '-ss';
$afterArguments[] = $from;
}
2018-07-03 18:09:45 +00:00
if (!empty($to)) {
2018-07-03 17:47:35 +00:00
if (!preg_match($durationRegex, $to)) {
throw new Exception(_('Invalid end time: ').$to.'.');
}
$afterArguments[] = '-to';
$afterArguments[] = $to;
}
2017-12-23 14:14:43 +00:00
$arguments = array_merge(
[
$this->config->avconv,
'-v', $this->config->avconvVerbosity,
],
$this->getRtmpArguments(),
2017-12-23 14:14:43 +00:00
[
'-i', $this->url,
'-f', $filetype,
'-b:a', $audioBitrate.'k',
],
2018-07-03 17:47:35 +00:00
$afterArguments,
[
2017-12-23 14:14:43 +00:00
'pipe:1',
]
);
//Vimeo needs a correct user-agent
$arguments[] = '-user_agent';
$arguments[] = $this->getProp('dump-user-agent');
return new Process($arguments);
2016-10-14 17:01:51 +00:00
}
2016-08-01 11:29:13 +00:00
/**
2016-09-07 22:28:28 +00:00
* Get audio stream of converted video.
2016-08-01 11:29:13 +00:00
*
2019-04-21 16:35:24 +00:00
* @param string $from Start the conversion at this time
* @param string $to End the conversion at this time
2016-08-01 11:29:13 +00:00
*
* @throws Exception If your try to convert an M3U8 video
* @throws Exception If the popen stream was not created correctly
2017-12-19 14:20:52 +00:00
*
* @return resource popen stream
2016-08-01 11:29:13 +00:00
*/
public function getAudioStream($from = null, $to = null)
{
if (isset($this->_type) && $this->_type == 'playlist') {
throw new Exception(_('Conversion of playlists is not supported.'));
}
if (isset($this->protocol)) {
if (in_array($this->protocol, ['m3u8', 'm3u8_native'])) {
2018-07-03 17:47:35 +00:00
throw new Exception(_('Conversion of M3U8 files is not supported.'));
} elseif ($this->protocol == 'http_dash_segments') {
2018-07-03 17:47:35 +00:00
throw new Exception(_('Conversion of DASH segments is not supported.'));
}
}
$avconvProc = $this->getAvconvProcess($this->config->audioBitrate, 'mp3', true, $from, $to);
2017-12-23 14:14:43 +00:00
$stream = popen($avconvProc->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new Exception(_('Could not open popen stream.'));
}
return $stream;
}
2016-12-26 14:50:26 +00:00
2016-12-26 14:58:07 +00:00
/**
* Get video stream from an M3U playlist.
*
* @throws Exception If avconv/ffmpeg is missing
* @throws Exception If the popen stream was not created correctly
2017-12-19 14:20:52 +00:00
*
* @return resource popen stream
2016-12-26 14:58:07 +00:00
*/
public function getM3uStream()
2016-12-26 14:50:26 +00:00
{
if (!$this->checkCommand([$this->config->avconv, '-version'])) {
throw new Exception(_('Can\'t find avconv or ffmpeg at ').$this->config->avconv.'.');
2016-12-26 14:50:26 +00:00
}
$process = new Process(
2016-12-26 14:50:26 +00:00
[
$this->config->avconv,
2017-12-23 14:14:43 +00:00
'-v', $this->config->avconvVerbosity,
'-i', $this->url,
'-f', $this->ext,
2016-12-26 14:50:26 +00:00
'-c', 'copy',
'-bsf:a', 'aac_adtstoasc',
'-movflags', 'frag_keyframe+empty_moov',
'pipe:1',
]
);
$stream = popen($process->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new Exception(_('Could not open popen stream.'));
}
return $stream;
2016-12-26 14:50:26 +00:00
}
2017-04-24 22:40:24 +00:00
/**
* Get an avconv stream to remux audio and video.
*
* @throws Exception If the popen stream was not created correctly
2017-12-19 14:20:52 +00:00
*
* @return resource popen stream
2017-04-24 22:40:24 +00:00
*/
public function getRemuxStream()
2017-04-24 22:40:24 +00:00
{
$urls = $this->getUrl();
if (!isset($urls[0]) || !isset($urls[1])) {
throw new Exception(_('This video does not have two URLs.'));
}
$process = new Process(
2017-04-24 22:40:24 +00:00
[
$this->config->avconv,
2017-12-23 14:14:43 +00:00
'-v', $this->config->avconvVerbosity,
2017-04-24 22:40:24 +00:00
'-i', $urls[0],
'-i', $urls[1],
'-c', 'copy',
'-map', '0:v:0 ',
'-map', '1:a:0',
'-f', 'matroska',
'pipe:1',
]
);
2017-04-24 22:41:49 +00:00
$stream = popen($process->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new Exception(_('Could not open popen stream.'));
}
return $stream;
2017-04-24 22:40:24 +00:00
}
/**
* Get video stream from an RTMP video.
*
* @throws Exception If the popen stream was not created correctly
2017-12-19 14:20:52 +00:00
*
* @return resource popen stream
*/
public function getRtmpStream()
{
2017-12-23 14:14:43 +00:00
$process = new Process(
array_merge(
[
$this->config->avconv,
'-v', $this->config->avconvVerbosity,
],
$this->getRtmpArguments(),
2017-12-23 14:14:43 +00:00
[
'-i', $this->url,
'-f', $this->ext,
2017-12-23 14:14:43 +00:00
'pipe:1',
]
)
);
$stream = popen($process->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new Exception(_('Could not open popen stream.'));
}
return $stream;
}
2017-05-02 15:04:55 +00:00
/**
* Get the stream of a converted video.
*
* @param int $audioBitrate Audio bitrate of the converted file
* @param string $filetype Filetype of the converted file
*
* @throws Exception If your try to convert and M3U8 video
* @throws Exception If the popen stream was not created correctly
*
* @return resource popen stream
*/
public function getConvertedStream($audioBitrate, $filetype)
{
if (in_array($this->protocol, ['m3u8', 'm3u8_native'])) {
2018-02-06 18:11:57 +00:00
throw new Exception(_('Conversion of M3U8 files is not supported.'));
}
$avconvProc = $this->getAvconvProcess($audioBitrate, $filetype, false);
$stream = popen($avconvProc->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new Exception(_('Could not open popen stream.'));
}
return $stream;
}
/**
* Get the same video but with another format.
*
* @param string $format New format
*
* @return Video
*/
public function withFormat($format)
{
2019-04-21 16:35:24 +00:00
return new self($this->webpageUrl, $format, $this->password);
}
/**
* Get a HTTP response containing the video.
*
* @return Response
*/
public function getHttpResponse()
{
$client = new Client();
$urls = $this->getUrl();
return $client->request('GET', $urls[0], ['stream' => true]);
}
2014-03-13 19:07:56 +00:00
}