Use ffmpeg instead of rtmpdump
This commit is contained in:
parent
7940446af3
commit
f8c8935b4c
11 changed files with 57 additions and 122 deletions
|
@ -2,7 +2,7 @@
|
||||||
install:
|
install:
|
||||||
- sc config wuauserv start= auto
|
- sc config wuauserv start= auto
|
||||||
- net start wuauserv
|
- net start wuauserv
|
||||||
- cinst php composer ffmpeg rtmpdump
|
- cinst php composer ffmpeg
|
||||||
- refreshenv
|
- refreshenv
|
||||||
- copy C:\tools\php72\php.ini-development C:\tools\php72\php.ini
|
- copy C:\tools\php72\php.ini-development C:\tools\php72\php.ini
|
||||||
- echo extension=C:\tools\php72\ext\php_gmp.dll >> C:\tools\php72\php.ini
|
- echo extension=C:\tools\php72\ext\php_gmp.dll >> C:\tools\php72\php.ini
|
||||||
|
|
|
@ -130,14 +130,14 @@ server {
|
||||||
|
|
||||||
## Other dependencies
|
## Other dependencies
|
||||||
|
|
||||||
You need [avconv](https://libav.org/avconv.html) and [rtmpdump](http://rtmpdump.mplayerhq.hu/)
|
You need [avconv](https://libav.org/avconv.html)
|
||||||
in order to enable conversions.
|
in order to enable conversions.
|
||||||
If you don't want to enable conversions, you can disable it in `config.yml`.
|
If you don't want to enable conversions, you can disable it in `config.yml`.
|
||||||
|
|
||||||
On Debian-based systems:
|
On Debian-based systems:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
sudo apt-get install libav-tools rtmpdump
|
sudo apt-get install libav-tools
|
||||||
```
|
```
|
||||||
|
|
||||||
You also probably need to edit the `avconv` variable in `config.yml`
|
You also probably need to edit the `avconv` variable in `config.yml`
|
||||||
|
|
|
@ -54,13 +54,6 @@ class Config
|
||||||
*/
|
*/
|
||||||
public $avconv = 'vendor/bin/ffmpeg';
|
public $avconv = 'vendor/bin/ffmpeg';
|
||||||
|
|
||||||
/**
|
|
||||||
* rtmpdump binary path.
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
public $rtmpdump = 'vendor/bin/rtmpdump';
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Disable URL rewriting.
|
* Disable URL rewriting.
|
||||||
*
|
*
|
||||||
|
@ -111,7 +104,6 @@ class Config
|
||||||
* * youtubedl: youtube-dl binary path
|
* * youtubedl: youtube-dl binary path
|
||||||
* * python: Python binary path
|
* * python: Python binary path
|
||||||
* * avconv: avconv or ffmpeg binary path
|
* * avconv: avconv or ffmpeg binary path
|
||||||
* * rtmpdump: rtmpdump binary path
|
|
||||||
* * params: Array of youtube-dl parameters
|
* * params: Array of youtube-dl parameters
|
||||||
* * convert: Enable conversion?
|
* * convert: Enable conversion?
|
||||||
*
|
*
|
||||||
|
|
|
@ -201,54 +201,32 @@ class VideoDownload
|
||||||
*
|
*
|
||||||
* @param object $video Video object returned by youtube-dl
|
* @param object $video Video object returned by youtube-dl
|
||||||
*
|
*
|
||||||
* @return array
|
* @return array Arguments
|
||||||
*/
|
*/
|
||||||
private function getRtmpProcessArguments(\stdClass $video)
|
private function getRtmpArguments(\stdClass $video)
|
||||||
{
|
{
|
||||||
$arguments = [
|
|
||||||
$this->config->rtmpdump,
|
|
||||||
'-q',
|
|
||||||
];
|
|
||||||
foreach ([
|
foreach ([
|
||||||
'url' => 'rtmp',
|
'url' => '-rtmp_tcurl',
|
||||||
'webpage_url' => 'pageUrl',
|
'webpage_url' => '-rtmp_pageurl',
|
||||||
'player_url' => 'swfVfy',
|
'player_url' => '-rtmp_swfverify',
|
||||||
'flash_version' => 'flashVer',
|
'flash_version' => '-rtmp_flashver',
|
||||||
'play_path' => 'playpath',
|
'play_path' => '-rtmp_playpath',
|
||||||
'app' => 'app',
|
'app' => '-rtmp_app',
|
||||||
] as $property => $option) {
|
] as $property => $option) {
|
||||||
if (isset($video->{$property})) {
|
if (isset($video->{$property})) {
|
||||||
$arguments[] = '--'.$option;
|
$arguments[] = $option;
|
||||||
$arguments[] = $video->{$property};
|
$arguments[] = $video->{$property};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $arguments;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get a process that runs rtmp in order to download a video.
|
|
||||||
*
|
|
||||||
* @param object $video Video object returned by youtube-dl
|
|
||||||
*
|
|
||||||
* @throws \Exception If rtmpdump is missing
|
|
||||||
*
|
|
||||||
* @return \Symfony\Component\Process\Process Process
|
|
||||||
*/
|
|
||||||
private function getRtmpProcess(\stdClass $video)
|
|
||||||
{
|
|
||||||
if (!$this->checkCommand([$this->config->rtmpdump, '--help'])) {
|
|
||||||
throw(new \Exception('Can\'t find rtmpdump'));
|
|
||||||
}
|
|
||||||
$arguments = $this->getRtmpProcessArguments($video);
|
|
||||||
if (isset($video->rtmp_conn)) {
|
if (isset($video->rtmp_conn)) {
|
||||||
foreach ($video->rtmp_conn as $conn) {
|
foreach ($video->rtmp_conn as $conn) {
|
||||||
$arguments[] = '--conn';
|
$arguments[] = '-rtmp_conn';
|
||||||
$arguments[] = $conn;
|
$arguments[] = $conn;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return new Process($arguments);
|
return $arguments;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -269,28 +247,39 @@ class VideoDownload
|
||||||
/**
|
/**
|
||||||
* Get a process that runs avconv in order to convert a video to MP3.
|
* Get a process that runs avconv in order to convert a video to MP3.
|
||||||
*
|
*
|
||||||
* @param string $url URL of the video file
|
* @param object $url Video object returned by youtube-dl
|
||||||
*
|
*
|
||||||
* @throws \Exception If avconv/ffmpeg is missing
|
* @throws \Exception If avconv/ffmpeg is missing
|
||||||
*
|
*
|
||||||
* @return \Symfony\Component\Process\Process Process
|
* @return \Symfony\Component\Process\Process Process
|
||||||
*/
|
*/
|
||||||
private function getAvconvMp3Process($url)
|
private function getAvconvMp3Process(\stdClass $video)
|
||||||
{
|
{
|
||||||
if (!$this->checkCommand([$this->config->avconv, '-version'])) {
|
if (!$this->checkCommand([$this->config->avconv, '-version'])) {
|
||||||
throw(new \Exception('Can\'t find avconv or ffmpeg'));
|
throw(new \Exception('Can\'t find avconv or ffmpeg'));
|
||||||
}
|
}
|
||||||
|
|
||||||
$arguments = [
|
if ($video->protocol == 'rtmp') {
|
||||||
$this->config->avconv,
|
$rtmpArguments = $this->getRtmpArguments($video);
|
||||||
'-v', $this->config->avconvVerbosity,
|
} else {
|
||||||
'-i', $url,
|
$rtmpArguments = [];
|
||||||
'-f', 'mp3',
|
}
|
||||||
'-b:a', $this->config->audioBitrate.'k',
|
|
||||||
'-vn',
|
$arguments = array_merge(
|
||||||
'pipe:1',
|
[
|
||||||
];
|
$this->config->avconv,
|
||||||
if ($url != '-') {
|
'-v', $this->config->avconvVerbosity,
|
||||||
|
],
|
||||||
|
$rtmpArguments,
|
||||||
|
[
|
||||||
|
'-i', $video->url,
|
||||||
|
'-f', 'mp3',
|
||||||
|
'-b:a', $this->config->audioBitrate.'k',
|
||||||
|
'-vn',
|
||||||
|
'pipe:1',
|
||||||
|
]
|
||||||
|
);
|
||||||
|
if ($video->url != '-') {
|
||||||
//Vimeo needs a correct user-agent
|
//Vimeo needs a correct user-agent
|
||||||
$arguments[] = '-user_agent';
|
$arguments[] = '-user_agent';
|
||||||
$arguments[] = $this->getProp(null, null, 'dump-user-agent');
|
$arguments[] = $this->getProp(null, null, 'dump-user-agent');
|
||||||
|
@ -318,17 +307,9 @@ class VideoDownload
|
||||||
throw(new \Exception('Conversion of M3U8 files is not supported.'));
|
throw(new \Exception('Conversion of M3U8 files is not supported.'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (parse_url($video->url, PHP_URL_SCHEME) == 'rtmp') {
|
$avconvProc = $this->getAvconvMp3Process($video);
|
||||||
$process = $this->getRtmpProcess($video);
|
|
||||||
$chain = new Chain($process);
|
|
||||||
$chain->pipe($this->getAvconvMp3Process('-'));
|
|
||||||
|
|
||||||
$stream = popen($chain->getProcess()->getCommandLine(), 'r');
|
$stream = popen($avconvProc->getCommandLine(), 'r');
|
||||||
} else {
|
|
||||||
$avconvProc = $this->getAvconvMp3Process($video->url);
|
|
||||||
|
|
||||||
$stream = popen($avconvProc->getCommandLine(), 'r');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!is_resource($stream)) {
|
if (!is_resource($stream)) {
|
||||||
throw new \Exception('Could not open popen stream.');
|
throw new \Exception('Could not open popen stream.');
|
||||||
|
@ -356,7 +337,7 @@ class VideoDownload
|
||||||
$process = new Process(
|
$process = new Process(
|
||||||
[
|
[
|
||||||
$this->config->avconv,
|
$this->config->avconv,
|
||||||
'-v', 'quiet',
|
'-v', $this->config->avconvVerbosity,
|
||||||
'-i', $video->url,
|
'-i', $video->url,
|
||||||
'-f', $video->ext,
|
'-f', $video->ext,
|
||||||
'-c', 'copy',
|
'-c', 'copy',
|
||||||
|
@ -388,7 +369,7 @@ class VideoDownload
|
||||||
$process = new Process(
|
$process = new Process(
|
||||||
[
|
[
|
||||||
$this->config->avconv,
|
$this->config->avconv,
|
||||||
'-v', 'quiet',
|
'-v', $this->config->avconvVerbosity,
|
||||||
'-i', $urls[0],
|
'-i', $urls[0],
|
||||||
'-i', $urls[1],
|
'-i', $urls[1],
|
||||||
'-c', 'copy',
|
'-c', 'copy',
|
||||||
|
@ -418,7 +399,21 @@ class VideoDownload
|
||||||
*/
|
*/
|
||||||
public function getRtmpStream(\stdClass $video)
|
public function getRtmpStream(\stdClass $video)
|
||||||
{
|
{
|
||||||
$stream = popen($this->getRtmpProcess($video)->getCommandLine(), 'r');
|
$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)) {
|
if (!is_resource($stream)) {
|
||||||
throw new \Exception('Could not open popen stream.');
|
throw new \Exception('Could not open popen stream.');
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,7 +25,6 @@
|
||||||
"doctrine/instantiator": "~1.0.0",
|
"doctrine/instantiator": "~1.0.0",
|
||||||
"ffmpeg/ffmpeg": "dev-release",
|
"ffmpeg/ffmpeg": "dev-release",
|
||||||
"rg3/youtube-dl": "2017.12.10",
|
"rg3/youtube-dl": "2017.12.10",
|
||||||
"rudloff/rtmpdump-bin": "~2.3.0",
|
|
||||||
"heroku/heroku-buildpack-php": "*"
|
"heroku/heroku-buildpack-php": "*"
|
||||||
},
|
},
|
||||||
"extra": {
|
"extra": {
|
||||||
|
|
30
composer.lock
generated
30
composer.lock
generated
|
@ -4,7 +4,7 @@
|
||||||
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file",
|
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#composer-lock-the-lock-file",
|
||||||
"This file is @generated automatically"
|
"This file is @generated automatically"
|
||||||
],
|
],
|
||||||
"content-hash": "81616b497a5848eeb2aafb7b7dcfbed3",
|
"content-hash": "96a0790e383a8870104483939363bcc0",
|
||||||
"packages": [
|
"packages": [
|
||||||
{
|
{
|
||||||
"name": "aura/session",
|
"name": "aura/session",
|
||||||
|
@ -2162,34 +2162,6 @@
|
||||||
},
|
},
|
||||||
"type": "library"
|
"type": "library"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"name": "rudloff/rtmpdump-bin",
|
|
||||||
"version": "2.3.1",
|
|
||||||
"source": {
|
|
||||||
"type": "git",
|
|
||||||
"url": "https://github.com/Rudloff/rtmpdump-bin.git",
|
|
||||||
"reference": "1042538e0a97940c6605c67c26ff48932ba5e61f"
|
|
||||||
},
|
|
||||||
"dist": {
|
|
||||||
"type": "zip",
|
|
||||||
"url": "https://api.github.com/repos/Rudloff/rtmpdump-bin/zipball/1042538e0a97940c6605c67c26ff48932ba5e61f",
|
|
||||||
"reference": "1042538e0a97940c6605c67c26ff48932ba5e61f",
|
|
||||||
"shasum": ""
|
|
||||||
},
|
|
||||||
"require-dev": {
|
|
||||||
"rtmpdump/rtmpdump": "2.3"
|
|
||||||
},
|
|
||||||
"bin": [
|
|
||||||
"rtmpdump"
|
|
||||||
],
|
|
||||||
"type": "library",
|
|
||||||
"notification-url": "https://packagist.org/downloads/",
|
|
||||||
"license": [
|
|
||||||
"GPL-2.0"
|
|
||||||
],
|
|
||||||
"description": "rtmpdump binary for Linux 64 bit",
|
|
||||||
"time": "2017-12-23T12:11:49+00:00"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"name": "sebastian/code-unit-reverse-lookup",
|
"name": "sebastian/code-unit-reverse-lookup",
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
|
|
|
@ -21,9 +21,6 @@ avconv: vendor/bin/ffmpeg
|
||||||
# avconv/ffmpeg logging level.
|
# avconv/ffmpeg logging level.
|
||||||
avconvVerbosity: error
|
avconvVerbosity: error
|
||||||
|
|
||||||
# Path to your rtmpdump binary
|
|
||||||
rtmpdump: vendor/bin/rtmpdump
|
|
||||||
|
|
||||||
# True to disable URL rewriting
|
# True to disable URL rewriting
|
||||||
uglyUrls: false
|
uglyUrls: false
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,6 @@ convert: false
|
||||||
python: C:\Python36\python.exe
|
python: C:\Python36\python.exe
|
||||||
avconv: C:\ProgramData\chocolatey\bin\ffmpeg.exe
|
avconv: C:\ProgramData\chocolatey\bin\ffmpeg.exe
|
||||||
avconvVerbosity: fatal
|
avconvVerbosity: fatal
|
||||||
rtmpdump: C:\ProgramData\chocolatey\bin\rtmpdump
|
|
||||||
youtubedl: C:\Python36\Lib\site-packages\youtube_dl\__main__.py
|
youtubedl: C:\Python36\Lib\site-packages\youtube_dl\__main__.py
|
||||||
params:
|
params:
|
||||||
- --no-warnings
|
- --no-warnings
|
||||||
|
|
|
@ -18,7 +18,6 @@ Here are the parameters that you can set:
|
||||||
* `params`: an array of parameters to pass to youtube-dl
|
* `params`: an array of parameters to pass to youtube-dl
|
||||||
* `convert`: true to enable audio conversion
|
* `convert`: true to enable audio conversion
|
||||||
* `avconv`: path to your avconv or ffmpeg binary
|
* `avconv`: path to your avconv or ffmpeg binary
|
||||||
* `rtmpdump`: path to your rtmpdump binary
|
|
||||||
* `remux`: enable remux mode (experimental)
|
* `remux`: enable remux mode (experimental)
|
||||||
|
|
||||||
See [`config.example.yml`](../config/config.example.yml) for default values.
|
See [`config.example.yml`](../config/config.example.yml) for default values.
|
||||||
|
|
|
@ -62,7 +62,6 @@ class ConfigTest extends TestCase
|
||||||
$this->assertInternalType('string', $config->youtubedl);
|
$this->assertInternalType('string', $config->youtubedl);
|
||||||
$this->assertInternalType('string', $config->python);
|
$this->assertInternalType('string', $config->python);
|
||||||
$this->assertInternalType('string', $config->avconv);
|
$this->assertInternalType('string', $config->avconv);
|
||||||
$this->assertInternalType('string', $config->rtmpdump);
|
|
||||||
$this->assertInternalType('bool', $config->convert);
|
$this->assertInternalType('bool', $config->convert);
|
||||||
$this->assertInternalType('bool', $config->uglyUrls);
|
$this->assertInternalType('bool', $config->uglyUrls);
|
||||||
$this->assertInternalType('bool', $config->stream);
|
$this->assertInternalType('bool', $config->stream);
|
||||||
|
|
|
@ -385,23 +385,6 @@ class VideoDownloadTest extends TestCase
|
||||||
$download->getAudioStream($url, $format);
|
$download->getAudioStream($url, $format);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Test getAudioStream function without curl or rtmpdump.
|
|
||||||
*
|
|
||||||
* @param string $url URL
|
|
||||||
* @param string $format Format
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
* @expectedException Exception
|
|
||||||
* @dataProvider rtmpUrlProvider
|
|
||||||
*/
|
|
||||||
public function testGetAudioStreamRtmpError($url, $format)
|
|
||||||
{
|
|
||||||
$this->config->rtmpdump = 'foobar';
|
|
||||||
$download = new VideoDownload($this->config);
|
|
||||||
$download->getAudioStream($url, $format);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test getAudioStream function with a M3U8 file.
|
* Test getAudioStream function with a M3U8 file.
|
||||||
*
|
*
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue