Implemented FFmpeg progress listener

This commit is contained in:
Ziver Koc 2021-01-31 23:26:46 +01:00
parent 7f7b8d3ab3
commit e9886a7afb
3 changed files with 309 additions and 1 deletions

View file

@ -25,6 +25,7 @@
package zutil.osal.app.ffmpeg;
import zutil.osal.app.ffmpeg.FFmpegConstants.*;
import zutil.osal.app.ffmpeg.FFmpegProgressManager.FFmpegProgressListener;
import java.util.ArrayList;
import java.util.List;
@ -40,6 +41,7 @@ public class FFmpeg {
private boolean overwriteOutput = false;
private List<FFmpegInput> inputs = new ArrayList<>();
private List<FFmpegOutput> outputs = new ArrayList<>();
private FFmpegProgressManager progressManager;
public FFmpeg() {}
@ -65,6 +67,16 @@ public class FFmpeg {
}
public void setProgressListener(FFmpegProgressListener listener) {
if (listener == null)
throw new IllegalArgumentException("FFmpegProgressListener cannot be NULL.");
if (progressManager != null)
progressManager.close();
progressManager = new FFmpegProgressManager(listener);
}
public String buildCommand() {
StringBuilder command = new StringBuilder();
command.append("ffmpeg");
@ -75,11 +87,14 @@ public class FFmpeg {
command.append(" -loglevel ").append(logLevel.toString().toLowerCase());
}
if (progressManager != null) {
command.append(" -progress ").append(progressManager.getAddress());
}
if (overwriteOutput) {
command.append(" -y");
}
// TODO: -progress url (global) for progress status
// TODO: -stdin Enable interaction on standard input
// Inputs

View file

@ -0,0 +1,219 @@
/*
* The MIT License (MIT)
*
* Copyright (c) 2021 Ziver Koc
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package zutil.osal.app.ffmpeg;
import zutil.log.LogUtil;
import zutil.net.threaded.ThreadedTCPNetworkServer;
import zutil.net.threaded.ThreadedTCPNetworkServerThread;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.InetAddress;
import java.net.Socket;
import java.util.logging.Level;
import java.util.logging.Logger;
public class FFmpegProgressManager extends ThreadedTCPNetworkServer {
private static final Logger logger = LogUtil.getLogger();
private static final int PROGRESS_DEFAULT_PORT = 5697;
/**
* A interface where FFmpeg progress will be reported.
*/
public interface FFmpegProgressListener {
/**
* Method will be called everytime there is new information about the progress of the FFmpeg execution.
*
* @param progress Object containing progress information.
*/
void ffmpegProgress(FFmpegProgress progress);
}
/**
* A Data class containing FFmpeg progress information.
*/
public static class FFmpegProgress {
public long frame = -1;
public float fps = -1;
public int bitrate = -1;
public long total_size = -1;
public long out_time_us = -1;
public long out_time_ms = -1;
public long out_time = -1;
public long dup_frames = -1;
public int drop_frames = -1;
public float speed = -1;
public FFmpegProgressStatus progress;
}
public enum FFmpegProgressStatus {
CONTINUE,
END;
public static FFmpegProgressStatus valueOfIgnoreCase(String status) {
for (FFmpegProgressStatus s : FFmpegProgressStatus.values()) {
if (s.name().equalsIgnoreCase(status))
return s;
}
throw new IllegalArgumentException("Unknown progress status: " + status);
}
}
// ----------------------------------------------------
// Manager variables
// ----------------------------------------------------
private FFmpegProgressListener listener;
private String address;
public FFmpegProgressManager(FFmpegProgressListener listener) {
this(listener, PROGRESS_DEFAULT_PORT);
}
public FFmpegProgressManager(FFmpegProgressListener listener, int port) {
super(port);
this.listener = listener;
this.address = "tcp://" + InetAddress.getLoopbackAddress() + ":" + port;
}
@Override
protected ThreadedTCPNetworkServerThread getThreadInstance(Socket s) throws IOException {
return new FFmpegProgressParserThread(s, listener);
}
/**
* @return The address where the progress information should be pushed to.
*/
public String getAddress() {
return address;
}
public FFmpegProgressListener getListener() {
return listener;
}
protected static class FFmpegProgressParserThread implements ThreadedTCPNetworkServerThread {
private Socket socket;
private BufferedReader in;
private FFmpegProgressListener listener;
protected FFmpegProgressParserThread(Socket s, FFmpegProgressListener listener) throws IOException {
this.socket = s;
this.listener = listener;
this.in = new BufferedReader(new InputStreamReader(s.getInputStream()));
}
@Override
public void run() {
try {
FFmpegProgress progress = new FFmpegProgress();
String line;
while ((line = in.readLine()) != null) {
parseProgress(line, progress);
if (progress.progress != null) {
// End of FFmpeg progress reporting, so report it and reset data object
listener.ffmpegProgress(progress);
progress = new FFmpegProgress();
}
}
} catch (IOException e) {
logger.log(Level.SEVERE, "FFmpeg progress parser thread crashed.", e);
}
try {
in.close();
} catch (IOException e) {
logger.log(Level.SEVERE, "FFmpeg progress parser thread crashed closing socket.", e);
}
}
protected static void parseProgress(String line, FFmpegProgress progress) {
String[] args = line.split("=", 2);
if (args.length != 2)
return;
String key = args[0].trim();
String value = args[1].trim();
if (value.equals("N/A"))
return;
switch (key) {
case "frame":
progress.frame = Long.parseLong(value);
break;
case "fps":
progress.fps = Float.parseFloat(value);
break;
case "bitrate":
progress.bitrate = Integer.parseInt(value);
break;
case "total_size":
progress.total_size = Long.parseLong(value);
break;
case "out_time_us": // microseconds
progress.out_time_us = Long.parseLong(value);
break;
case "out_time_ms": // milliseconds
progress.out_time_ms = Long.parseLong(value);
break;
// TODO: out_time=00:01:47.000000
case "dup_frames":
progress.dup_frames = Long.parseLong(value);
break;
case "drop_frames":
progress.drop_frames = Integer.parseInt(value);
break;
case "speed":
value = value.replace("x", "");
progress.speed = Float.parseFloat(value);
break;
case "progress":
progress.progress = FFmpegProgressStatus.valueOfIgnoreCase(value);
break;
default:
// TODO: Handle stream information, e.g. stream_0_0_xxx=xxx:
break;
}
}
}
}

View file

@ -0,0 +1,74 @@
/*
* The MIT License (MIT)
*
* Copyright (c) 2021 Ziver Koc
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
package zutil.osal.app.ffmpeg;
import org.junit.Test;
import zutil.osal.app.ffmpeg.FFmpegProgressManager.FFmpegProgressParserThread;
import static org.junit.Assert.*;
import static zutil.osal.app.ffmpeg.FFmpegProgressManager.FFmpegProgressStatus;
public class FFmpegProgressManagerTest {
@Test
public void simpleParsing() {
FFmpegProgressManager.FFmpegProgress progress = new FFmpegProgressManager.FFmpegProgress();
FFmpegProgressParserThread.parseProgress("frame=2675", progress);
FFmpegProgressParserThread.parseProgress("fps=0.00", progress);
FFmpegProgressParserThread.parseProgress("bitrate=3000", progress);
FFmpegProgressParserThread.parseProgress("total_size=10000", progress);
FFmpegProgressParserThread.parseProgress("out_time_us=107000000", progress);
FFmpegProgressParserThread.parseProgress("out_time_ms=107000000", progress);
FFmpegProgressParserThread.parseProgress("out_time=00:01:47.000000", progress);
FFmpegProgressParserThread.parseProgress("dup_frames=0", progress);
FFmpegProgressParserThread.parseProgress("drop_frames=0", progress);
FFmpegProgressParserThread.parseProgress("speed=214x", progress);
FFmpegProgressParserThread.parseProgress("progress=continue", progress);
assertEquals(2675, progress.frame);
assertEquals(0.0f, progress.fps, 0.001);
assertEquals(3000, progress.bitrate);
assertEquals(10000, progress.total_size);
assertEquals(107000000, progress.out_time_us);
assertEquals(107000000, progress.out_time_ms);
assertEquals(0, progress.dup_frames);
assertEquals(0, progress.drop_frames);
assertEquals(214f, progress.speed, 0.001);
assertEquals(FFmpegProgressStatus.CONTINUE, progress.progress);
}
public void parsingNA() {
FFmpegProgressManager.FFmpegProgress progress = new FFmpegProgressManager.FFmpegProgress();
FFmpegProgressParserThread.parseProgress("bitrate=N/A", progress);
FFmpegProgressParserThread.parseProgress("total_size=N/A", progress);
FFmpegProgressParserThread.parseProgress("progress=end", progress);
assertEquals(3000, progress.bitrate);
assertEquals(10000, progress.total_size);
assertEquals(FFmpegProgressStatus.END, progress.progress);
}
}