Tornando contenuto binario da un'azione JPF con Weblogic Portal 10.2
-
13-09-2019 - |
Domanda
Una delle azioni di mio controller JPF costruisce un file PDF e vorrei tornare questo file per l'utente in modo che possa scaricarlo.
E 'possibile farlo o sono costretto a scrivere il file da qualche parte e hanno la mia azione in avanti un link a questo file? Si noti che vorrei evitare che, per quanto possibile, per ragioni di sicurezza e perché non ho modo di sapere quando l'utente ha scaricato il file in modo che posso eliminarlo.
Ho cercato di accedere al HttpServletResponse, ma non succede nulla:
getResponse().setContentLength(file.getSize());
getResponse().setContentType(file.getMimeType());
getResponse().setHeader("Content-Disposition", "attachment;filename=\"" + file.getTitle() + "\"");
getResponse().getOutputStream().write(file.getContent());
getResponse().flushBuffer();
Soluzione
Abbiamo qualcosa di simile, tranne tornando immagini al posto di un PDF; dovrebbe essere una soluzione simile, però, sto cercando di indovinare.
In una JSP, abbiamo un tag IMG
, dove il src
è impostato su:
<c:url value="/path/getImage.do?imageId=${imageID}" />
(non sto mostrando tutto, perché sto cercando di semplificare.) Nel tuo caso, forse sarebbe un collegamento, in cui il href
è fatto in un modo simile.
che mappa getImage.do
al nostro controller JPF, ovviamente. Ecco il codice dal metodo getImage()
JPF, che è la parte che si sta cercando di lavorare su:
@Jpf.Action(forwards = {
@Jpf.Forward(name = FWD_SUCCESS, navigateTo = Jpf.NavigateTo.currentPage),
@Jpf.Forward(name = FWD_FAILURE, navigateTo = Jpf.NavigateTo.currentPage) })
public Forward getImage(final FormType pForm) throws Exception {
final HttpServletRequest lRequest = getRequest();
final HttpServletResponse lResponse = getResponse();
final HttpSession lHttpSession = getSession();
final String imageIdParam = lRequest.getParameter("imageId");
final long header = lRequest.getDateHeader("If-Modified-Since");
final long current = System.currentTimeMillis();
if (header > 0 && current - header < MAX_AGE_IN_SECS * 1000) {
lResponse.setStatus(HttpServletResponse.SC_NOT_MODIFIED);
return null;
}
try {
if (imageIdParam == null) {
throw new IllegalArgumentException("imageId is null.");
}
// Call to EJB, which is retrieving the image from
// a separate back-end system
final ImageType image = getImage(lHttpSession, Long
.parseLong(imageIdParam));
if (image == null) {
lResponse.sendError(404, IMAGE_DOES_NOT_EXIST);
return null;
}
lResponse.setContentType(image.getType());
lResponse.addDateHeader("Last-Modified", current);
// public: Allows authenticated responses to be cached.
lResponse.setHeader("Cache-Control", "max-age=" + MAX_AGE_IN_SECS
+ ", public");
lResponse.setHeader("Expires", null);
lResponse.setHeader("Pragma", null);
lResponse.getOutputStream().write(image.getContent());
} catch (final IllegalArgumentException e) {
LogHelper.error(this.getClass(), "Illegal argument.", e);
lResponse.sendError(404, IMAGE_DOES_NOT_EXIST);
} catch (final Exception e) {
LogHelper.error(this.getClass(), "General exception.", e);
lResponse.sendError(500);
}
return null;
}
In realtà ho tolto molto poco da questo metodo, perché c'è molto poco in là che ho bisogno di nascondere da occhi indiscreti - il codice è piuttosto generica, interessati con le immagini, non con la logica di business. (Ho cambiato alcuni dei nomi dei tipi di dati, ma niente di grave.)