I have the following code in a java Web Service:
public boolean makeFile(String fileName, String audio)
{
if (makeUserFolder())
{
File file = new File(getUserFolderPath() + fileName + amr);
FileOutputStream fileOutputStream = null;
try
{
file.createNewFile();
fileOutputStream = new FileOutputStream(file);
fileOutputStream.write(Base64.decode(audio));
return true;
}
catch(FileNotFoundException ex)
{
return false;
}
catch(IOException ex)
{
return false;
}
finally{
try {
fileOutputStream.close();
convertFile(fileName);
} catch (IOException ex) {
Logger.getLogger(FileUtils.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
else
return false;
}
public boolean convertFile(String fileName)
{
Process ffmpeg;
String filePath = this.userFolderPath + fileName;
try {
ProcessBuilder pb = new ProcessBuilder("ffmpeg","-i",filePath + amr,filePath + mp3);
pb.redirectErrorStream();
ffmpeg = pb.start();
} catch (IOException ex) {
return false;
}
return true;
}
It used to work and now it simply won't execute the ffmpeg conversion for some reason. I thought it was a problem with my file but after running the command from terminal no errors are thrown or anything, thought it was maybe permissions issue but all the permissions have been granted in the folder I'm saving the files. I noticed that the input BufferedReader ins being set to null after running the process, any idea what's happening?
First of all, a small nitpick with your code...when you create the
FileOutputStream
you create it using a string rather than aFile
, when you have already created theFile
before, so you might as well recycle that rather than force theFileOutputStream
to instantiate theFile
itself.Another small nitpick is the fact that when you are writing out the audio file, you should enclose that in a
try
block and close the output stream in afinally
block. If you are allowed to add a new library to your project, you might use Guava which has a methodFiles.write(byte[],File)
, which will take care of all the dirty resource management for you.The only thing that I can see that looks like a definite bug is the fact that you are ignoring the error stream of ffmpeg. If you are blocking waiting for input on the stdout of ffmpeg, then it will not work.
The easiest way to take care of this bug is to use
ProcessBuilder
instead ofRuntime
.If you start it this way, then your current code will be able to read both input streams fully. It is possible that the stderr was hiding some error that you were not able to see due to not reading it.
If that was not your problem, I would recommend using absolute paths with ffmpeg...in other words:
If that doesn't work, I would change ffmpeg to be an absolute path as well (in order to rule out path issues).
Edit: Further suggestions.
I would personally refactor the writing code into its own method, so that you can use it elsewhere necessary. In other other words:
The reason that I made the
closeQuietly(Closeable)
method is due to the fact that if you do not close it in that way, there is a possibility that an exception will be thrown by theclose()
method, and that exception will obscure the exception that was thrown originally. If you put these in a utility class (although looking at your code, I assume that the class that it is currently in is named FileUtils), then you will be able to use them throughout your application whenever you need to deal with file output.This will allow you to rewrite the block as:
I don't know whether or not you should do this, however if you want to be sure that the ffmpeg process has completed, then you should say
ffmpeg.waitFor();
to be sure that it has completed. If you do that, then you should examineffmpeg.exitValue();
to make sure that it completed successfully.Another thing that you might want to do is once it has completed, write what it output to a log file so you have a record of what happened, just in case something happens.