You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
The GraphicImageManager leaks files upon refresh of a page with <p:graphicImage>.
Essentially the problem seems to be that on the first page rendering, where the functionality works properly, the StreamContent of the view access scoped bean gets closed and its contets saved on a temp file.
On a susequent request to the page (e.g. refresh of the labrador page) the the uniqueId of the page stay the same and the code would just ask for the operating system to go ahead and create a new temp file. The new temp file would be a leakage since it corresponded to an already existing unique id and in addition it would be empty because the original StreamedContent input stream had already been closed on the first rendereing.
I refactored the register method to be as follows:
/**
\* During the rendering phase of p:graphic image for which the avanced graphic image feature is enabled the
\* StreamedContent gets writtent to a temporary file in the OS tmp file folder. Finally, the uniqueId will be used
\* to do a lookup of path to the temporary file to render the iamge requested.
*
\* @param content
\* the primefaces streamed content for an image which should be written out to an OS temporary file.
\* @param uniqueId
\* a unique id for the current user session and image access, used to create a new unique temp file
*
*/
public void registerImage(StreamedContent content, String uniqueId) {
// (1) Trivial scenario - there is nothing to be done here because
// the user is most likely doing a page refresh and the image already exists in a temp file location
if (storedContent.containsKey(uniqueId)) {
LOGGER.log(Level.FINEST,
"The streamed content for uniqueId will not be saved to a new file since it already exists in the cache. UiqueId: "
+ uniqueId);
return;
}
// (2) The uniqueId is brand new one so the streamed cotent streamed has not yet been closed
// and we can save the data to a temporary file.
ReadableByteChannel inputChannel = null;
WritableByteChannel outputChannel = null;
try {
File tempFile = File.createTempFile(uniqueId, "primefaces");
storedContent.put(uniqueId, tempFile.getAbsolutePath());
// get a channel from the stream
inputChannel = Channels.newChannel(content.getStream());
outputChannel = Channels.newChannel(new FileOutputStream(tempFile));
// copy the channels
fastChannelCopy(inputChannel, outputChannel);
} catch (IOException e) {
LOGGER.log(
Level.SEVERE,
"Unexpected error took place while attempting to save primefaces streamed content image to the temporary fold",
e);
} finally {
// closing the channels
try {
if (inputChannel != null) {
inputChannel.close();
}
} catch (Exception e) {
LOGGER.log(
Level.SEVERE,
"Unexpected error took place while attempting to close the input stream of the image to be saved into a temporary location",
e);
}
try {
if (outputChannel != null) {
outputChannel.close();
}
} catch (Exception e) {
LOGGER.log(Level.SEVERE,
"Unexpected error took place while attempting to close the output stream of the temp file", e);
}
}
}
Thanks
The text was updated successfully, but these errors were encountered:
I can't see how this method can be called more then once with the same uniqueId parameter value.
Since this value is made unique by the call to the method org.primefaces.component.graphicimage.GraphicImageRenderer#createUniqueContentId which generates this uniqueId.
The GraphicImageManager leaks files upon refresh of a page with <p:graphicImage>.
Essentially the problem seems to be that on the first page rendering, where the functionality works properly, the StreamContent of the view access scoped bean gets closed and its contets saved on a temp file.
On a susequent request to the page (e.g. refresh of the labrador page) the the uniqueId of the page stay the same and the code would just ask for the operating system to go ahead and create a new temp file. The new temp file would be a leakage since it corresponded to an already existing unique id and in addition it would be empty because the original StreamedContent input stream had already been closed on the first rendereing.
I refactored the register method to be as follows:
/** \* During the rendering phase of p:graphic image for which the avanced graphic image feature is enabled the \* StreamedContent gets writtent to a temporary file in the OS tmp file folder. Finally, the uniqueId will be used \* to do a lookup of path to the temporary file to render the iamge requested. * \* @param content \* the primefaces streamed content for an image which should be written out to an OS temporary file. \* @param uniqueId \* a unique id for the current user session and image access, used to create a new unique temp file * */ public void registerImage(StreamedContent content, String uniqueId) {
Thanks
The text was updated successfully, but these errors were encountered: