Правильно напишите HTML-страницу в ответе сервлета.
У меня есть сервлет, развернутый в http://ip:8080/simple
Сервлет в упаковке a.b.c
У меня есть HTML-страница в a.b.resources
названный Test.html
,
HTML имеет img
тег для изображения.
В сервлете я делаю:
htmlFile = MyServlet.class.getResourceAsStream("/a/b/resources/Test.html");
resp.setContentType("text/html");
PrintWriter writer = resp.getWriter();
byte[] bytes=new byte[htmlFile.available()];
htmlFile.read(bytes);
resp.setContentLength(bytes.length);
writer.print(new String(bytes));
writer.flush();
writer.close();
HTML-страница появляется в браузере, но вместо изображения я вижу ее alt
описание.
Я пытался:
<img alt="Company A" src="./CompanyLogo.jpg">
<img alt="Company A" src="/a/b/resources/CompanyLogo.jpg">
<img alt="Company A" src="CompanyLogo.jpg">
Но ни одна из этих работ.
Изображение jpg находится в / a / b / c / resources, то есть в том же каталоге, что и HTML-страница.
Я использую встроенный Jetty.
Что я тут возую?
3 ответа
Браузер пытается разрешить эти ресурсы относительно текущего URI запроса (как вы видите в адресной строке браузера). Эти ресурсы, конечно же, не существуют в вашем общедоступном веб-контенте, поскольку вы, кажется, разместили их в пути к классам.
Чтобы решить эту проблему, вам действительно нужно проанализировать HTML-код и изменить все, относящиеся к домену. src
и / или href
атрибуты <a>
, <img>
, <base>
, <link>
, <script>
, <iframe>
элементы etc, чтобы они указывали на сервлет, который передает эти ресурсы из пути к классам в ответ HTTP.
Это немного работы, но Jsoup делает это легко. Вот пример, который предполагает, что ваш сервлет сопоставлен с шаблоном URL /proxy/*
,
String proxyURL = request.getContextPath() + "/proxy/";
InputStream input = MyServlet.class.getResourceAsStream("/a/b/resources" + request.getPathInfo());
if (request.getRequestURI().endsWith(".html")) { // A HTML page is been requested.
Document document = Jsoup.parse(input, "UTF-8", null);
for (Element element : document.select("[href]")) {
element.attr("href", proxyURL + element.attr("href"));
}
for (Element element : document.select("[src]")) {
element.attr("src", proxyURL + element.attr("src"));
}
response.setContentType("text/html;charset=UTF-8");
response.setCharacterEncoding("UTF-8");
response.getWriter().write(document.html());
}
else { // Other resources like images, etc which have been proxied to this servlet.
response.setContentType(getServletContext().getMimeType(request.getPathInfo()));
OutputStream output = response.getOutputStream();
byte[] buffer = new byte[8192];
for (int length = 0; (length = input.read(buffer)) > 0;) {
output.write(buffer, 0, length);
}
}
input.close();
Откройте его по http://yourdomain:yourport/contextname/proxy/test.html.
Нет способа сделать это без реализации сервлета, который будет считывать изображение из файла ресурсов. Попробуй это:
public void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException
{
byte[] bbuf = new byte[8192];
resp.setContentType(req.getSession().getServletContext().getMimeType( req.getPathInfo()));
InputStream in = MyImageServlet.class.getResourceAsStream("/"+req.getPathInfo());
OutputStream op = resp.getOutputStream();
int length;
while ((in != null) && ((length = in.read(bbuf)) != -1)){
op.write(bbuf,0,length);
op.flush();
}
in.close();
op.close();
}
затем зарегистрируйте его в своем web.xml
вот так
<servlet-mapping>
<servlet-name>fetchimage</servlet-name>
<url-pattern>/fetchimage/*</url-pattern>
</servlet-mapping>
а затем использовать его так
<img alt="Company A" src="/fetchimage/a/b/resources/CompanyLogo.jpg">
Вам нужно будет реализовать много проверок на ошибки (МНОГО ПРОВЕРКИ ОШИБОК, просто чтобы уточнить:)), отфильтровать пути, чтобы удостовериться, что кто-то не может просто читать ваши файлы классов, используя ту же технику, но некоторые вариации на этом должен работать на вас.
request.getRequestDispatcher("/a/b/Test.html").forward(запрос, ответ);