package net.knarcraft.ffmpegconverter.utility; import net.knarcraft.ffmpegconverter.streams.AudioStream; import net.knarcraft.ffmpegconverter.streams.StreamObject; import net.knarcraft.ffmpegconverter.streams.SubtitleStream; import net.knarcraft.ffmpegconverter.streams.VideoStream; import java.io.BufferedReader; import java.io.File; import java.io.IOException; import java.io.InputStreamReader; import java.util.ArrayList; import java.util.Arrays; import java.util.List; /** * A class which helps with ffmpeg probing and converting */ public final class FFMpegHelper { private static final String PROBE_SPLIT_CHARACTER = "øæåÆØå"; private FFMpegHelper() { } /** * Gets streams from a file * * @param ffprobePath
The path/command to ffprobe.
* @param fileThe file to probe.
* @returnA list of StreamObjects.
* @throws IOExceptionIf the process can't be readProcess.
*/ public static ListThe executable to use (ffmpeg/ffprobe).
* @param fileNameThe name of the file to execute on.
* @returnA base list of ffmpeg commands for converting a video for web
*/ public static ListThe executable to use (ffmpeg/ffprobe).
* @param fileNameThe name of the file to execute on.
* @returnA base list of ffmpeg commands for converting a file.
*/ public static ListThe list containing the command to run.
* @param startThe offset before converting.
* @param lengthThe offset for stopping the conversion.
*/ public static void addDebugArguments(ListThe process to run.
* @param folderThe folder the process should run in.
* @param spacerThe character(s) to use between each new line read.
* @param writeWhether to write the output directly instead of storing it.
* @throws IOExceptionIf the process can't be readProcess.
*/ public static String runProcess(ProcessBuilder processBuilder, File folder, String spacer, boolean write) throws IOException { //Give the user information about what's about to happen OutputUtil.print("Command to be run: "); OutputUtil.println(processBuilder.command().toString()); //Set directory and error stream processBuilder.directory(folder); processBuilder.redirectErrorStream(true); Process process = processBuilder.start(); BufferedReader processReader = new BufferedReader(new InputStreamReader(process.getInputStream())); StringBuilder output = new StringBuilder(); while (process.isAlive()) { String read = readProcess(processReader, spacer); if (!read.equals("")) { if (write) { OutputUtil.println(read); } else { OutputUtil.printDebug(read); output.append(read); } } } OutputUtil.println("Process finished."); return output.toString(); } /** * Adds audio to a command * * @param commandThe command to add audio to.
* @param audioStreamThe audio stream to be added.
* @param toStereoWhether to convert the audio stream to stereo.
*/ public static void addAudioStreams(ListThe list containing the rest of the command.
* @param subtitleStreamThe subtitle stream to be used.
* @param videoStreamThe video stream to be used.
* @param fileThe file to convert.
*/ public static void addSubtitlesAndVideo(ListThe list containing the FFmpeg commands.
* @param subtitleStreamThe subtitle stream to add.
* @param videoStreamThe video stream to burn the subtitle into.
*/ private static void addSubtitle(ListThe filename to escape.
* @returnA filename with known special characters escaped.
*/ private static String escapeSpecialCharactersInFileName(String fileName) { return fileName.replaceAll("\\\\", "\\\\\\\\\\\\\\\\") .replaceAll("'", "'\\\\\\\\\\\\\''") .replaceAll("%", "\\\\\\\\\\\\%") .replaceAll(":", "\\\\\\\\\\\\:") .replace("]", "\\]") .replace("[", "\\["); } /** * Adds image subtitle commands to a command list * * @param commandThe list containing the FFmpeg commands.
* @param subtitleStreamThe subtitle stream to add.
* @param videoStreamThe video stream to burn the subtitle into.
*/ private static void addInternalImageSubtitle(ListThe list containing the FFmpeg commands.
* @param externalImageSubtitleThe external image subtitle stream to add.
* @param videoStreamThe video stream to burn the subtitle into.
*/ private static void addExternalImageSubtitle(ListThe path/command to ffprobe.
* @param fileThe file to probe.
* @returnA list of streams.
* @throws IOExceptionIf something goes wrong while probing.
*/ private static String[] probeForStreams(String ffprobePath, File file) throws IOException { ProcessBuilder processBuilder = new ProcessBuilder( ffprobePath, "-v", "error", "-show_entries", "stream_tags=language,title:stream=index,codec_name,codec_type,channels,codec_type,width,height", file.toString() ); String result = runProcess(processBuilder, file.getParentFile(), PROBE_SPLIT_CHARACTER, false); return StringUtil.stringBetween(result, "[STREAM]", "[/STREAM]"); } /** * Takes a list of all streams and parses each stream into one of three objects * * @param streamsA list of all streams for the current file.
* @param fileThe file currently being converted.
* @returnA list of StreamObjects.
*/ private static ListThe path/command to ffprobe.
* @param directoryThe directory containing the file.
* @param convertingFileThe file to be converted.
* @returnThe extension of the subtitle or empty if no subtitle was found.
*/ private static ListThe reader of a process.
* @returnThe output from the readProcess.
* @throws IOExceptionOn reader failure.
*/ private static String readProcess(BufferedReader reader, String spacer) throws IOException { String line; StringBuilder text = new StringBuilder(); while (reader.ready() && (line = reader.readLine()) != null && !line.equals("") && !line.equals("\n")) { text.append(line).append(spacer); } return text.toString().trim(); } /** * Parses a list of video stream parameters to a video stream object * * @param streamPartsA list of parameters belonging to an video stream.
* @param relativeIndexThe relative index of the video stream.
* @returnA SubtitleStream object.
* @throws NumberFormatExceptionIf codec index contains a non-numeric value.
*/ private static VideoStream parseVideoStream(String[] streamParts, int relativeIndex) throws NumberFormatException { String codec = null; int absoluteIndex = -1; int width = -1; int height = -1; for (String streamPart : streamParts) { if (streamPart.startsWith("codec_name=")) { codec = streamPart.replace("codec_name=", ""); } else if (streamPart.startsWith("index=")) { absoluteIndex = Integer.parseInt(streamPart.replace("index=", "")); } else if (streamPart.startsWith("width=")) { width = Integer.parseInt(streamPart.replace("width=", "")); } else if (streamPart.startsWith("height=")) { height = Integer.parseInt(streamPart.replace("height=", "")); } } return new VideoStream(codec, absoluteIndex, relativeIndex, width, height); } /** * Parses a list of audio stream parameters to an audio stream object * * @param streamPartsA list of parameters belonging to an audio stream.
* @param relativeIndexThe relative index of the audio stream.
* @returnA SubtitleStream object.
* @throws NumberFormatExceptionIf codec index contains a non-numeric value.
*/ private static AudioStream parseAudioStream(String[] streamParts, int relativeIndex) throws NumberFormatException { String codec = null; int absoluteIndex = -1; String language = null; int channels = 0; String title = ""; for (String streamPart : streamParts) { if (streamPart.startsWith("codec_name=")) { codec = streamPart.replace("codec_name=", ""); } else if (streamPart.startsWith("index=")) { absoluteIndex = Integer.parseInt(streamPart.replace("index=", "")); } else if (streamPart.startsWith("TAG:language=")) { language = streamPart.replace("TAG:language=", ""); } else if (streamPart.startsWith("channels=")) { channels = Integer.parseInt(streamPart.replace("channels=", "")); } else if (streamPart.startsWith("TAG:title=")) { title = streamPart.replace("TAG:title=", ""); } } return new AudioStream(codec, absoluteIndex, relativeIndex, language, title, channels); } /** * Parses a list of subtitle stream parameters to a subtitle stream object * * @param streamPartsA list of parameters belonging to a subtitle stream.
* @param relativeIndexThe relative index of the subtitle.
* @param fileThe file currently being converted.
* @returnA SubtitleStream object.
* @throws NumberFormatExceptionIf codec index contains a non-numeric value.
*/ private static SubtitleStream parseSubtitleStream(String[] streamParts, int relativeIndex, String file) throws NumberFormatException { String codecName = null; int absoluteIndex = -1; String language = null; String title = ""; for (String streamPart : streamParts) { if (streamPart.startsWith("codec_name=")) { codecName = streamPart.replace("codec_name=", ""); } else if (streamPart.startsWith("index=")) { absoluteIndex = Integer.parseInt(streamPart.replace("index=", "")); } else if (streamPart.startsWith("TAG:language=")) { language = streamPart.replace("TAG:language=", ""); } else if (streamPart.startsWith("TAG:title=")) { title = streamPart.replace("TAG:title=", ""); } } return new SubtitleStream(codecName, absoluteIndex, relativeIndex, language, title, file); } }