alltube/classes/VideoDownload.php

499 lines
15 KiB
PHP
Raw Normal View History

2014-03-13 20:07:56 +01:00
<?php
/**
2016-09-08 00:28:28 +02:00
* VideoDownload class.
2016-08-01 13:29:13 +02:00
*/
2016-12-05 13:12:27 +01:00
2015-10-29 20:43:43 +01:00
namespace Alltube;
2016-03-30 01:49:08 +02:00
use Symfony\Component\Process\Process;
2016-04-01 00:42:28 +02:00
2014-03-13 20:07:56 +01:00
/**
2016-09-08 00:28:28 +02:00
* Extract info about videos.
2016-08-01 13:29:13 +02:00
*/
2016-03-30 01:49:08 +02:00
class VideoDownload
2014-03-13 20:07:56 +01:00
{
2016-10-14 02:40:33 +02:00
/**
2016-10-14 02:40:57 +02:00
* Config instance.
*
2016-10-14 02:40:33 +02:00
* @var Config
*/
2016-10-10 21:32:07 +02:00
private $config;
2016-10-14 02:40:33 +02:00
2016-08-01 13:29:13 +02:00
/**
2016-09-08 00:28:28 +02:00
* VideoDownload constructor.
2017-12-19 15:20:52 +01:00
*
2017-12-24 01:12:47 +01:00
* @param Config $config Config instance.
*
2017-12-19 15:20:52 +01:00
* @throws \Exception If youtube-dl is missing
* @throws \Exception If Python is missing
2016-08-01 13:29:13 +02:00
*/
public function __construct(Config $config = null)
2016-04-08 19:06:41 +02:00
{
if (isset($config)) {
$this->config = $config;
} else {
$this->config = Config::getInstance();
}
/*
We don't translate these exceptions because they always occur before Slim can catch them
so they will always go to the logs.
*/
if (!is_file($this->config->youtubedl)) {
throw new \Exception("Can't find youtube-dl at ".$this->config->youtubedl);
2017-11-12 15:14:59 +01:00
} elseif (!$this->checkCommand([$this->config->python, '--version'])) {
2016-10-18 10:15:09 +02:00
throw new \Exception("Can't find Python at ".$this->config->python);
}
}
/**
2017-12-23 14:37:29 +01:00
* Return a youtube-dl process with the specified arguments.
*
* @param string[] $arguments Arguments
*
* @return Process
*/
private function getProcess(array $arguments)
{
return new Process(
array_merge(
2016-09-08 00:28:28 +02:00
[$this->config->python, $this->config->youtubedl],
$this->config->params,
$arguments
)
);
2016-04-08 19:06:41 +02:00
}
2014-03-13 20:07:56 +01:00
/**
2016-09-08 00:28:28 +02:00
* List all extractors.
*
2016-08-01 13:29:13 +02:00
* @return string[] Extractors
2014-03-13 20:07:56 +01:00
* */
2016-04-08 19:06:41 +02:00
public function listExtractors()
2014-03-13 20:07:56 +01:00
{
return explode("\n", trim($this->getProp(null, null, 'list-extractors')));
2014-03-13 20:07:56 +01:00
}
2016-10-14 02:40:33 +02:00
/**
2016-10-14 02:40:57 +02:00
* Get a property from youtube-dl.
*
* @param string $url URL to parse
* @param string $format Format
* @param string $prop Property
* @param string $password Video password
2016-10-14 02:40:57 +02:00
*
2017-12-19 15:20:52 +01: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
*
2016-10-14 02:40:33 +02:00
* @return string
*/
private function getProp($url, $format = null, $prop = 'dump-json', $password = null)
2014-03-13 20:07:56 +01:00
{
$arguments = [
'--'.$prop,
2017-12-23 14:37:29 +01:00
$url,
];
2014-03-13 20:07:56 +01:00
if (isset($format)) {
$arguments[] = '-f '.$format;
2014-03-13 20:07:56 +01:00
}
if (isset($password)) {
$arguments[] = '--video-password';
$arguments[] = $password;
}
$process = $this->getProcess($arguments);
//This is needed by the openload extractor because it runs PhantomJS
$process->setEnv(['PATH'=>$this->config->phantomjsDir]);
$process->inheritEnvironmentVariables();
2016-04-01 00:42:28 +02:00
$process->run();
if (!$process->isSuccessful()) {
$errorOutput = trim($process->getErrorOutput());
if ($errorOutput == 'ERROR: This video is protected by a password, use the --video-password option') {
throw new PasswordException($errorOutput);
} elseif (substr($errorOutput, 0, 21) == 'ERROR: Wrong password') {
throw new \Exception(_('Wrong password'));
} else {
throw new \Exception($errorOutput);
}
2014-03-18 15:08:16 +01:00
} else {
return trim($process->getOutput());
2014-03-18 15:08:16 +01:00
}
2014-03-13 20:07:56 +01:00
}
2016-10-13 16:40:19 +02:00
/**
* Get all information about a video.
*
* @param string $url URL of page
* @param string $format Format to use for the video
* @param string $password Video password
2016-10-13 16:40:19 +02:00
*
* @return object Decoded JSON
* */
public function getJSON($url, $format = null, $password = null)
2016-10-13 16:40:19 +02:00
{
2017-04-25 01:53:38 +02:00
return json_decode($this->getProp($url, $format, 'dump-single-json', $password));
2016-10-13 16:40:19 +02:00
}
2014-03-13 20:07:56 +01:00
/**
2016-09-08 00:28:28 +02:00
* Get URL of video from URL of page.
*
2017-04-25 00:40:24 +02:00
* It generally returns only one URL.
* But it can return two URLs when multiple formats are specified
* (eg. bestvideo+bestaudio).
*
* @param string $url URL of page
* @param string $format Format to use for the video
* @param string $password Video password
*
2017-04-25 14:55:21 +02:00
* @return string[] URLs of video
2014-03-13 20:07:56 +01:00
* */
public function getURL($url, $format = null, $password = null)
2014-03-13 20:07:56 +01:00
{
return explode("\n", $this->getProp($url, $format, 'get-url', $password));
2014-03-13 20:07:56 +01:00
}
2016-08-01 13:29:13 +02:00
/**
2016-09-08 00:28:28 +02:00
* Get filename of video file from URL of page.
2016-08-01 13:29:13 +02:00
*
* @param string $url URL of page
* @param string $format Format to use for the video
* @param string $password Video password
2016-08-01 13:29:13 +02:00
*
* @return string Filename of extracted video
* */
public function getFilename($url, $format = null, $password = null)
{
return trim($this->getProp($url, $format, 'get-filename', $password));
}
2016-08-01 13:29:13 +02:00
/**
2017-04-25 00:41:49 +02:00
* Get filename of video with the specified extension.
2016-08-01 13:29:13 +02:00
*
2017-04-25 00:40:24 +02:00
* @param string $extension New file extension
* @param string $url URL of page
* @param string $format Format to use for the video
* @param string $password Video password
2016-08-01 13:29:13 +02:00
*
2017-04-25 00:40:24 +02:00
* @return string Filename of extracted video with specified extension
*/
public function getFileNameWithExtension($extension, $url, $format = null, $password = null)
{
return html_entity_decode(
pathinfo(
$this->getFilename($url, $format, $password),
PATHINFO_FILENAME
2017-04-25 00:40:24 +02:00
).'.'.$extension,
ENT_COMPAT,
'ISO-8859-1'
);
}
2017-04-25 00:40:24 +02:00
/**
* Get filename of audio from URL of page.
*
* @param string $url URL of page
* @param string $format Format to use for the video
* @param string $password Video password
*
* @return string Filename of converted audio file
* */
public function getAudioFilename($url, $format = null, $password = null)
{
return $this->getFileNameWithExtension('mp3', $url, $format, $password);
}
2016-10-15 16:18:04 +02:00
/**
* Return arguments used to run rtmp for a specific video.
2016-10-15 16:18:04 +02:00
*
* @param object $video Video object returned by youtube-dl
2016-10-15 16:18:04 +02:00
*
2017-12-23 15:14:43 +01:00
* @return array Arguments
2016-10-15 16:18:04 +02:00
*/
2017-12-23 15:14:43 +01:00
private function getRtmpArguments(\stdClass $video)
2016-10-15 16:18:04 +02:00
{
2017-12-23 16:04:55 +01:00
$arguments = [];
2016-10-15 16:18:04 +02:00
foreach ([
2017-12-23 15:14:43 +01:00
'url' => '-rtmp_tcurl',
'webpage_url' => '-rtmp_pageurl',
'player_url' => '-rtmp_swfverify',
'flash_version' => '-rtmp_flashver',
'play_path' => '-rtmp_playpath',
'app' => '-rtmp_app',
2016-10-15 16:18:04 +02:00
] as $property => $option) {
if (isset($video->{$property})) {
2017-12-23 15:14:43 +01:00
$arguments[] = $option;
2017-12-23 14:37:29 +01:00
$arguments[] = $video->{$property};
2016-10-15 16:18:04 +02:00
}
}
2016-10-15 16:20:54 +02:00
2016-10-14 19:01:51 +02:00
if (isset($video->rtmp_conn)) {
foreach ($video->rtmp_conn as $conn) {
2017-12-23 15:14:43 +01:00
$arguments[] = '-rtmp_conn';
2017-12-23 14:37:29 +01:00
$arguments[] = $conn;
2016-10-14 19:01:51 +02:00
}
}
2016-10-14 19:02:14 +02:00
2017-12-23 15:14:43 +01:00
return $arguments;
2016-10-14 19:01:51 +02:00
}
/**
2017-05-14 00:54:47 +02:00
* Check if a command runs successfully.
*
* @param array $command Command and arguments
*
* @return bool False if the command returns an error, true otherwise
*/
private function checkCommand(array $command)
{
$process = new Process($command);
$process->run();
return $process->isSuccessful();
}
2016-10-14 19:01:51 +02:00
/**
* Get a process that runs avconv in order to convert a video.
2016-10-14 19:02:14 +02:00
*
* @param object $video Video object returned by youtube-dl
* @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
2016-10-14 19:02:14 +02:00
*
2017-12-19 15:20:52 +01:00
* @throws \Exception If avconv/ffmpeg is missing
*
2017-12-24 01:12:47 +01:00
* @return Process Process
2016-10-14 19:01:51 +02:00
*/
private function getAvconvProcess(\stdClass $video, $audioBitrate, $filetype = 'mp3', $audioOnly = true)
2016-10-14 19:01:51 +02:00
{
if (!$this->checkCommand([$this->config->avconv, '-version'])) {
throw(new \Exception(_('Can\'t find avconv or ffmpeg.')));
2016-10-14 19:01:51 +02:00
}
2016-10-14 19:02:14 +02:00
2017-12-23 15:14:43 +01:00
if ($video->protocol == 'rtmp') {
$rtmpArguments = $this->getRtmpArguments($video);
} else {
$rtmpArguments = [];
}
if ($audioOnly) {
$videoArguments = ['-vn'];
} else {
$videoArguments = [];
}
2017-12-23 15:14:43 +01:00
$arguments = array_merge(
[
$this->config->avconv,
'-v', $this->config->avconvVerbosity,
],
$rtmpArguments,
[
'-i', $video->url,
'-f', $filetype,
'-b:a', $audioBitrate.'k',
],
$videoArguments,
[
2017-12-23 15:14:43 +01:00
'pipe:1',
]
);
if ($video->url != '-') {
//Vimeo needs a correct user-agent
$arguments[] = '-user_agent';
$arguments[] = $this->getProp(null, null, 'dump-user-agent');
}
return new Process($arguments);
2016-10-14 19:01:51 +02:00
}
2016-08-01 13:29:13 +02:00
/**
2016-09-08 00:28:28 +02:00
* Get audio stream of converted video.
2016-08-01 13:29:13 +02:00
*
* @param string $url URL of page
* @param string $format Format to use for the video
* @param string $password Video password
2016-08-01 13:29:13 +02:00
*
2017-12-19 15:20:52 +01:00
* @throws \Exception If your try to convert and M3U8 video
* @throws \Exception If the popen stream was not created correctly
*
* @return resource popen stream
2016-08-01 13:29:13 +02:00
*/
public function getAudioStream($url, $format, $password = null)
{
$video = $this->getJSON($url, $format, $password);
if (in_array($video->protocol, ['m3u8', 'm3u8_native'])) {
throw(new \Exception(_('Conversion of M3U8 files is not supported.')));
}
$avconvProc = $this->getAvconvProcess($video, $this->config->audioBitrate);
2017-12-23 15:14:43 +01:00
$stream = popen($avconvProc->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new \Exception(_('Could not open popen stream.'));
}
return $stream;
}
2016-12-26 15:50:26 +01:00
2016-12-26 15:58:07 +01:00
/**
* Get video stream from an M3U playlist.
*
* @param \stdClass $video Video object returned by getJSON
*
2017-12-19 15:20:52 +01:00
* @throws \Exception If avconv/ffmpeg is missing
* @throws \Exception If the popen stream was not created correctly
*
* @return resource popen stream
2016-12-26 15:58:07 +01:00
*/
2016-12-26 15:50:26 +01:00
public function getM3uStream(\stdClass $video)
{
if (!$this->checkCommand([$this->config->avconv, '-version'])) {
throw(new \Exception(_('Can\'t find avconv or ffmpeg.')));
2016-12-26 15:50:26 +01:00
}
$process = new Process(
2016-12-26 15:50:26 +01:00
[
$this->config->avconv,
2017-12-23 15:14:43 +01:00
'-v', $this->config->avconvVerbosity,
2016-12-26 15:50:26 +01:00
'-i', $video->url,
'-f', $video->ext,
'-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 15:50:26 +01:00
}
2017-04-25 00:40:24 +02:00
/**
* Get an avconv stream to remux audio and video.
*
* @param array $urls URLs of the video ($urls[0]) and audio ($urls[1]) files
*
2017-12-19 15:20:52 +01:00
* @throws \Exception If the popen stream was not created correctly
*
* @return resource popen stream
2017-04-25 00:40:24 +02:00
*/
public function getRemuxStream(array $urls)
{
$process = new Process(
2017-04-25 00:40:24 +02:00
[
$this->config->avconv,
2017-12-23 15:14:43 +01:00
'-v', $this->config->avconvVerbosity,
2017-04-25 00:40:24 +02:00
'-i', $urls[0],
'-i', $urls[1],
'-c', 'copy',
'-map', '0:v:0 ',
'-map', '1:a:0',
'-f', 'matroska',
'pipe:1',
]
);
2017-04-25 00:41:49 +02:00
$stream = popen($process->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new \Exception(_('Could not open popen stream.'));
}
return $stream;
2017-04-25 00:40:24 +02:00
}
/**
* Get video stream from an RTMP video.
*
* @param \stdClass $video Video object returned by getJSON
*
2017-12-19 15:20:52 +01:00
* @throws \Exception If the popen stream was not created correctly
*
* @return resource popen stream
*/
public function getRtmpStream(\stdClass $video)
{
2017-12-23 15:14:43 +01:00
$process = new Process(
array_merge(
[
$this->config->avconv,
'-v', $this->config->avconvVerbosity,
],
$this->getRtmpArguments($video),
[
'-i', $video->url,
'-f', $video->ext,
'pipe:1',
]
)
);
$stream = popen($process->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new \Exception(_('Could not open popen stream.'));
}
return $stream;
}
2017-05-02 17:04:55 +02:00
/**
* Get a Tar stream containing every video in the playlist piped through the server.
*
2017-05-05 00:06:18 +02:00
* @param object $video Video object returned by youtube-dl
2017-05-02 17:04:55 +02:00
* @param string $format Requested format
*
2017-12-19 15:20:52 +01:00
* @throws \Exception If the popen stream was not created correctly
*
* @return resource
2017-05-02 17:04:55 +02:00
*/
2017-05-05 01:51:28 +02:00
public function getPlaylistArchiveStream(\stdClass $video, $format)
2017-05-02 17:04:55 +02:00
{
$playlistItems = [];
foreach ($video->entries as $entry) {
$playlistItems[] = urlencode($entry->url);
}
$stream = fopen('playlist://'.implode(';', $playlistItems).'/'.$format, 'r');
if (!is_resource($stream)) {
throw new \Exception(_('Could not open fopen stream.'));
}
2017-05-02 17:04:55 +02:00
return $stream;
}
/**
* Get the stream of a converted video.
*
* @param string $url URL of page
* @param string $format Source format to use for the conversion
* @param int $audioBitrate Audio bitrate of the converted file
* @param string $filetype Filetype of the converted file
* @param string $password Video password
*
* @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($url, $format, $audioBitrate, $filetype, $password = null)
{
$video = $this->getJSON($url, $format, $password);
if (in_array($video->protocol, ['m3u8', 'm3u8_native'])) {
throw(new \Exception(_('Conversion of M3U8 files is not supported.')));
}
$avconvProc = $this->getAvconvProcess($video, $audioBitrate, $filetype, false);
$stream = popen($avconvProc->getCommandLine(), 'r');
if (!is_resource($stream)) {
throw new \Exception(_('Could not open popen stream.'));
}
return $stream;
}
2014-03-13 20:07:56 +01:00
}