Je viens de commencer à développer des services REST, mais je suis tombé sur une situation difficile: envoyer des fichiers de mon service REST à mon client. Jusqu'à présent, j'ai compris comment envoyer des types de données simples (chaînes, entiers, etc.), mais l'envoi d'un fichier est une autre affaire car il y a tellement de formats de fichiers que je ne sais même pas par où commencer. Mon service REST est fait sur Java et j'utilise Jersey, j'envoie toutes les données au format JSON.
J'ai lu sur l'encodage base64, certaines personnes disent que c'est une bonne technique, d'autres disent que ce n'est pas à cause de problèmes de taille de fichier. Quelle est la bonne manière? Voici à quoi ressemble une classe de ressources simple dans mon projet:
import java.sql.SQLException;
import java.util.List;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
import javax.ws.rs.core.Context;
import javax.ws.rs.core.MediaType;
import javax.ws.rs.core.Request;
import javax.ws.rs.core.UriInfo;
import com.mx.ipn.escom.testerRest.dao.TemaDao;
import com.mx.ipn.escom.testerRest.modelo.Tema;
@Path("/temas")
public class TemaResource {
@GET
@Produces({MediaType.APPLICATION_JSON})
public List<Tema> getTemas() throws SQLException{
TemaDao temaDao = new TemaDao();
List<Tema> temas=temaDao.getTemas();
temaDao.terminarSesion();
return temas;
}
}
J'imagine que le code pour envoyer un fichier serait quelque chose comme:
import java.sql.SQLException;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.Produces;
@Path("/resourceFiles")
public class FileResource {
@GET
@Produces({application/x-octet-stream})
public File getFiles() throws SQLException{ //I'm not really sure what kind of data type I should return
// Code for encoding the file or just send it in a data stream, I really don't know what should be done here
return file;
}
}
Quel type d'annotations dois-je utiliser? J'ai vu certaines personnes recommander pour une @GET
utilisation @Produces({application/x-octet-stream})
, est-ce la bonne façon? Les fichiers que j'envoie sont spécifiques, le client n'a donc pas besoin de parcourir les fichiers. Quelqu'un peut-il me guider sur la manière dont je suis censé envoyer le fichier? Dois-je l'encoder en utilisant base64 pour l'envoyer en tant qu'objet JSON? ou l'encodage n'est pas nécessaire pour l'envoyer en tant qu'objet JSON? Merci pour toute aide que vous pourriez apporter.
java.io.File
(ou un chemin de fichier) sur votre serveur ou les données proviennent-elles d'une autre source, comme une base de données, un service Web, un appel de méthode renvoyant unInputStream
?Réponses:
Je ne recommande pas d'encoder les données binaires en base64 et de les encapsuler dans JSON. Cela augmentera inutilement la taille de la réponse et ralentira les choses.
Diffusez simplement vos données de fichier en utilisant GET et en
application/octect-stream
utilisant l'une des méthodes d'usine dejavax.ws.rs.core.Response
(une partie de l'API JAX-RS, vous n'êtes donc pas enfermé dans Jersey):Si vous n'avez pas d'
File
objet réel , mais unInputStream
,Response.ok(entity, mediaType)
devrait être capable de gérer cela également.la source
ZipOutputStream
long avec un retourStreamingOutput
degetFile()
. De cette façon, vous obtenez un format multi-fichiers bien connu que la plupart des clients devraient facilement pouvoir lire. N'utilisez la compression que si cela a du sens pour vos données, c'est-à-dire pas pour les fichiers précompressés comme les JPEG. Du côté client, il y aZipInputStream
pour analyser la réponse.Si vous souhaitez renvoyer un fichier à télécharger, spécialement si vous souhaitez l'intégrer à certaines bibliothèques javascript de téléchargement / téléchargement de fichiers, le code ci-dessous devrait faire le travail:
la source
Changez l'adresse de la machine d'hôte local en adresse IP à laquelle vous souhaitez que votre client se connecte pour appeler le service mentionné ci-dessous.
Client pour appeler le webservice REST:
Service au client de réponse:
JAR nécessaire:
WEB.XML:
la source
Puisque vous utilisez JSON, je l'encoderais en Base64 avant de l'envoyer sur le fil.
Si les fichiers sont volumineux, essayez de regarder BSON ou un autre format mieux adapté aux transferts binaires.
Vous pouvez également compresser les fichiers, s'ils sont bien compressés, avant de les encoder en base64.
la source
@Produces
contenir mon annotation?