ffmpeg-mappable-media/src/FFMpeg/FFMpeg.php

273 lines
7.8 KiB
PHP
Raw Normal View History

2012-04-13 10:20:54 +02:00
<?php
2012-04-13 14:34:53 +02:00
/*
* This file is part of PHP-FFmpeg.
*
* (c) Alchemy <info@alchemy.fr>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
2012-04-13 10:20:54 +02:00
namespace FFMpeg;
2012-05-25 16:21:16 +02:00
use FFMpeg\Exception\InvalidArgumentException;
use FFMpeg\Exception\LogicException;
use FFMpeg\Exception\RuntimeException;
use FFMpeg\Format\AudioFormat;
use FFMpeg\Format\VideoFormat;
use Symfony\Component\Process\Process;
2012-04-13 15:12:43 +02:00
/**
* FFMpeg driver
*
* @author Romain Neutron imprec@gmail.com
*/
2012-04-13 10:20:54 +02:00
class FFMpeg extends Binary
{
protected $pathfile;
2012-04-13 15:12:43 +02:00
/**
* Opens a file in order to be processed
*
2012-05-25 16:22:16 +02:00
* @param string $pathfile A pathfile
* @return \FFMpeg\FFMpeg
2012-05-25 16:21:16 +02:00
* @throws InvalidArgumentException
2012-04-13 15:12:43 +02:00
*/
2012-04-13 10:20:54 +02:00
public function open($pathfile)
{
2012-05-11 00:30:02 +02:00
if ( ! file_exists($pathfile)) {
2012-05-25 16:21:16 +02:00
$this->logger->addError(sprintf('FFmpeg failed to open %s', $pathfile));
2012-04-13 15:12:43 +02:00
2012-05-25 16:21:16 +02:00
throw new InvalidArgumentException(sprintf('File %s does not exists', $pathfile));
2012-04-13 10:20:54 +02:00
}
$this->logger->addInfo(sprintf('FFmpeg opens %s', $pathfile));
$this->pathfile = $pathfile;
return $this;
}
/**
* Close a file
*
* @return \FFMpeg\FFMpeg
*/
public function close()
{
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg closes %s', $this->pathfile));
$this->pathfile = null;
return $this;
2012-04-13 10:20:54 +02:00
}
2012-04-13 15:12:43 +02:00
/**
*
2012-05-25 16:22:16 +02:00
* @param integer $time The time in second where to take the snapshot
* @param string $output The pathfile where to write
* @return \FFMpeg\FFMpeg
2012-05-25 16:21:16 +02:00
* @throws RuntimeException
* @throws LogicException
2012-04-13 15:12:43 +02:00
*/
public function extractImage($time, $output)
2012-04-13 10:20:54 +02:00
{
2012-05-11 00:30:02 +02:00
if ( ! $this->pathfile) {
2012-05-25 16:21:16 +02:00
throw new LogicException('No file open');
2012-04-13 10:20:54 +02:00
}
$cmd = $this->binary
2012-05-11 00:30:02 +02:00
. ' -i ' . escapeshellarg($this->pathfile)
. ' -vframes 1 -ss ' . $time
. ' -f image2 ' . escapeshellarg($output);
2012-04-13 10:20:54 +02:00
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg executes command %s', $cmd));
2012-04-13 10:20:54 +02:00
2012-04-13 15:12:43 +02:00
$process = new Process($cmd);
2012-05-11 00:30:02 +02:00
try {
2012-04-13 15:12:43 +02:00
$process->run();
2012-05-11 00:30:02 +02:00
} catch (\RuntimeException $e) {
2012-04-13 15:12:43 +02:00
}
2012-04-13 10:20:54 +02:00
2012-05-11 00:30:02 +02:00
if ( ! $process->isSuccessful()) {
2012-05-25 16:21:16 +02:00
$this->logger->addError(sprintf('FFmpeg command failed : %s', $process->getErrorOutput()));
2012-04-13 10:20:54 +02:00
2012-04-13 15:12:43 +02:00
$this->cleanupTemporaryFile($output);
2012-04-13 10:20:54 +02:00
2012-05-25 16:21:16 +02:00
throw new RuntimeException('Failed to extract image');
2012-04-13 10:20:54 +02:00
}
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg command successful'));
2012-04-13 10:20:54 +02:00
return $this;
2012-04-13 10:20:54 +02:00
}
2012-04-13 15:12:43 +02:00
/**
* Encode the file to the specified format
*
2012-05-25 16:22:16 +02:00
* @param AudioFormat $format The output format
* @param string $outputPathfile The pathfile where to write
* @param integer $threads The number of threads to use
* @return \FFMpeg\FFMpeg
2012-05-25 16:21:16 +02:00
* @throws RuntimeException
* @throws LogicException
2012-04-13 15:12:43 +02:00
*/
2012-05-25 16:21:16 +02:00
public function encode(AudioFormat $format, $outputPathfile, $threads = 1)
2012-04-13 10:20:54 +02:00
{
2012-05-11 00:30:02 +02:00
if ( ! $this->pathfile) {
2012-05-25 16:21:16 +02:00
throw new LogicException('No file open');
2012-04-13 10:20:54 +02:00
}
$threads = max(min($threads, 64), 1);
2012-05-11 00:30:02 +02:00
switch (true) {
2012-05-25 16:21:16 +02:00
case $format instanceof VideoFormat:
$this->encodeVideo($format, $outputPathfile, $threads);
2012-04-13 14:15:56 +02:00
break;
default:
2012-05-25 16:21:16 +02:00
case $format instanceof AudioFormat:
$this->encodeAudio($format, $outputPathfile, $threads);
2012-04-13 14:15:56 +02:00
break;
}
return $this;
2012-04-13 14:15:56 +02:00
}
2012-04-13 15:12:43 +02:00
/**
* Encode to audio
*
2012-05-25 16:22:16 +02:00
* @param AudioFormat $format The output format
* @param string $outputPathfile The pathfile where to write
* @param integer $threads The number of threads to use
* @return \FFMpeg\FFMpeg
2012-05-25 16:21:16 +02:00
* @throws RuntimeException
2012-04-13 15:12:43 +02:00
*/
2012-05-25 16:21:16 +02:00
protected function encodeAudio(AudioFormat $format, $outputPathfile, $threads)
2012-04-13 14:15:56 +02:00
{
$cmd = $this->binary
2012-05-11 00:30:02 +02:00
. ' -y -i '
. escapeshellarg($this->pathfile)
. ' ' . $format->getExtraParams()
. ' -threads ' . $threads
. ' -acodec ' . $format->getAudioCodec()
. ' -ab ' . $format->getKiloBitrate() . 'k '
. ' -ac 2 -ar ' . $format->getAudioSampleRate()
. ' ' . escapeshellarg($outputPathfile);
2012-04-13 14:15:56 +02:00
2012-04-13 15:12:43 +02:00
$process = new Process($cmd);
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg executes command %s', $cmd));
2012-05-11 00:30:02 +02:00
try {
2012-04-13 15:12:43 +02:00
$process->run();
2012-05-11 00:30:02 +02:00
} catch (\RuntimeException $e) {
2012-04-13 15:12:43 +02:00
}
2012-04-13 14:15:56 +02:00
2012-05-11 00:30:02 +02:00
if ( ! $process->isSuccessful()) {
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg command failed'));
throw new RuntimeException(sprintf('Encoding failed : %s', $process->getErrorOutput()));
2012-04-13 14:15:56 +02:00
}
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg command successful'));
return $this;
2012-04-13 14:15:56 +02:00
}
2012-04-13 15:12:43 +02:00
/**
* Encode to video
*
2012-05-25 16:22:16 +02:00
* @param VideoFormat $format The output format
* @param string $outputPathfile The pathfile where to write
* @param integer $threads The number of threads to use
* @return \FFMpeg\FFMpeg
2012-05-25 16:21:16 +02:00
* @throws RuntimeException
2012-04-13 15:12:43 +02:00
*/
2012-05-25 16:21:16 +02:00
protected function encodeVideo(VideoFormat $format, $outputPathfile, $threads)
2012-04-13 14:15:56 +02:00
{
2012-04-13 10:20:54 +02:00
$cmd_part1 = $this->binary
2012-05-11 00:30:02 +02:00
. ' -y -i '
. escapeshellarg($this->pathfile) . ' '
. $format->getExtraParams() . ' ';
2012-04-13 10:20:54 +02:00
2012-05-25 18:24:37 +02:00
$cmd_part2 = '';
if ($format->getWidth() && $format->getHieght()) {
$cmd_part2 .= ' -s ' . $format->getWidth() . 'x' . $format->getHeight();
}
$cmd_part2 .= ' -r ' . $format->getFrameRate()
2012-05-11 00:30:02 +02:00
. ' -vcodec ' . $format->getVideoCodec()
. ' -b ' . $format->getKiloBitrate() . 'k -g 25 -bf 3'
. ' -threads ' . $threads
. ' -refs 6 -b_strategy 1 -coder 1 -qmin 10 -qmax 51 '
. ' -sc_threshold 40 -flags +loop -cmp +chroma'
. ' -me_range 16 -subq 7 -i_qfactor 0.71 -qcomp 0.6 -qdiff 4 '
. ' -trellis 1 -qscale 1 '
. '-acodec ' . $format->getAudioCodec() . ' -ab 92k ';
2012-04-13 10:20:54 +02:00
$tmpFile = new \SplFileInfo(tempnam(sys_get_temp_dir(), 'temp') . '.' . pathinfo($outputPathfile, PATHINFO_EXTENSION));
$passes = array();
$passes[] = $cmd_part1 . ' -pass 1 ' . $cmd_part2
2012-05-11 00:30:02 +02:00
. ' -an ' . escapeshellarg($tmpFile->getPathname());
2012-04-13 10:20:54 +02:00
$passes[] = $cmd_part1 . ' -pass 2 ' . $cmd_part2
2012-05-11 00:30:02 +02:00
. ' -ac 2 -ar 44100 ' . escapeshellarg($outputPathfile);
2012-04-13 10:20:54 +02:00
2012-04-13 15:12:43 +02:00
$process = null;
2012-05-11 00:30:02 +02:00
foreach ($passes as $pass) {
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg executes command %s', $pass));
2012-04-13 15:12:43 +02:00
$process = new Process($pass);
2012-04-13 10:20:54 +02:00
2012-05-11 00:30:02 +02:00
try {
2012-05-11 00:34:19 +02:00
$process->run();
2012-05-11 00:30:02 +02:00
} catch (\RuntimeException $e) {
2012-04-13 10:20:54 +02:00
break;
}
}
$this->cleanupTemporaryFile($tmpFile->getPathname());
$this->cleanupTemporaryFile(getcwd() . '/ffmpeg2pass-0.log');
$this->cleanupTemporaryFile(getcwd() . '/ffmpeg2pass-0.log.mbtree');
2012-05-11 00:30:02 +02:00
if ( ! $process->isSuccessful()) {
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg command failed'));
throw new RuntimeException(sprintf('Encoding failed : %s', $process->getErrorOutput()));
2012-04-13 10:20:54 +02:00
}
2012-05-25 16:21:16 +02:00
$this->logger->addInfo(sprintf('FFmpeg command successful'));
return $this;
2012-04-13 10:20:54 +02:00
}
2012-04-13 15:12:43 +02:00
/**
* Removes unnecessary file
*
* @param string $pathfile
*/
2012-04-13 10:20:54 +02:00
protected function cleanupTemporaryFile($pathfile)
{
2012-05-11 00:30:02 +02:00
if (file_exists($pathfile) && is_writable($pathfile)) {
2012-04-13 10:20:54 +02:00
unlink($pathfile);
}
}
2012-04-13 15:12:43 +02:00
/**
2012-05-25 16:21:16 +02:00
* {@inheritdoc}
2012-04-13 15:12:43 +02:00
*
* @return string
*/
2012-04-13 10:20:54 +02:00
protected static function getBinaryName()
{
2012-05-11 00:30:02 +02:00
return array('avconv', 'ffmpeg');
2012-04-13 10:20:54 +02:00
}
}