mirror of
https://github.com/octoleo/plantuml-server.git
synced 2024-12-22 08:48:54 +00:00
Enhanced version with short URL scheme, http headers to take advantage of the browser cache, and a new structure with JSP files.
This commit is contained in:
parent
3847a2e170
commit
85169a1baf
3
.gitignore
vendored
3
.gitignore
vendored
@ -1,3 +1,4 @@
|
||||
|
||||
/user.property
|
||||
/WEB-INF/classes
|
||||
/plantuml.war
|
||||
/.settings
|
||||
|
@ -1,21 +1,23 @@
|
||||
package net.sourceforge.plantuml.servlet;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.io.PrintWriter;
|
||||
import java.net.URL;
|
||||
import java.net.URLDecoder;
|
||||
import java.util.StringTokenizer;
|
||||
import java.util.regex.Matcher;
|
||||
import java.util.regex.Pattern;
|
||||
|
||||
import javax.servlet.RequestDispatcher;
|
||||
import javax.servlet.ServletException;
|
||||
import javax.servlet.http.HttpServlet;
|
||||
import javax.servlet.http.HttpServletRequest;
|
||||
import javax.servlet.http.HttpServletResponse;
|
||||
|
||||
import net.sourceforge.plantuml.SourceStringReader;
|
||||
import net.sourceforge.plantuml.StringUtils;
|
||||
import net.sourceforge.plantuml.code.Transcoder;
|
||||
import net.sourceforge.plantuml.code.TranscoderUtil;
|
||||
|
||||
import HTTPClient.CookieModule;
|
||||
import HTTPClient.HTTPConnection;
|
||||
import HTTPClient.HTTPResponse;
|
||||
@ -32,15 +34,14 @@ import HTTPClient.ParseException;
|
||||
*/
|
||||
public class PlantUmlServlet extends HttpServlet {
|
||||
|
||||
private static final Pattern startumlPattern = Pattern
|
||||
.compile("/\\w+/uml/startuml/(.*)");
|
||||
|
||||
private static final Pattern imagePattern = Pattern
|
||||
.compile("/\\w+/uml/image/(.*)");
|
||||
|
||||
private static final Pattern proxyPattern = Pattern
|
||||
.compile("/\\w+/uml/proxy/((\\d+)/)?(http://.*)");
|
||||
private static final Pattern startumlPattern = Pattern.compile("/\\w+/start/(.*)");
|
||||
private static final Pattern imagePattern = Pattern.compile("/\\w+/img/(.*)");
|
||||
private static final Pattern proxyPattern = Pattern.compile("/\\w+/proxy/((\\d+)/)?(http://.*)");
|
||||
private static final Pattern oldStartumlPattern = Pattern.compile("/\\w+/uml/startuml/(.*)");
|
||||
private static final Pattern oldImagePattern = Pattern.compile("/\\w+/uml/image/(.*)");
|
||||
private static final Pattern oldProxyPattern = Pattern.compile("/\\w+/uml/proxy/((\\d+)/)?(http://.*)");
|
||||
|
||||
@Override
|
||||
public void doGet(HttpServletRequest request, HttpServletResponse response)
|
||||
throws IOException, ServletException {
|
||||
|
||||
@ -48,7 +49,9 @@ public class PlantUmlServlet extends HttpServlet {
|
||||
Matcher startumlMatcher = startumlPattern.matcher(uri);
|
||||
Matcher imageMatcher = imagePattern.matcher(uri);
|
||||
Matcher proxyMatcher = proxyPattern.matcher(uri);
|
||||
|
||||
Matcher oldStartumlMatcher = oldStartumlPattern.matcher(uri);
|
||||
Matcher oldImageMatcher = oldImagePattern.matcher(uri);
|
||||
Matcher oldProxyMatcher = oldProxyPattern.matcher(uri);
|
||||
if (startumlMatcher.matches()) {
|
||||
String source = startumlMatcher.group(1);
|
||||
handleImage(response, source);
|
||||
@ -59,79 +62,51 @@ public class PlantUmlServlet extends HttpServlet {
|
||||
String num = proxyMatcher.group(2);
|
||||
String source = proxyMatcher.group(3);
|
||||
handleImageProxy(response, num, source);
|
||||
} else if (oldStartumlMatcher.matches()) {
|
||||
String source = oldStartumlMatcher.group(1);
|
||||
handleImage(response, source);
|
||||
} else if (oldImageMatcher.matches()) {
|
||||
String source = oldImageMatcher.group(1);
|
||||
handleImageDecompress(response, source);
|
||||
} else if (oldProxyMatcher.matches()) {
|
||||
String num = oldProxyMatcher.group(2);
|
||||
String source = oldProxyMatcher.group(3);
|
||||
handleImageProxy(response, num, source);
|
||||
} else {
|
||||
doPost(request, response);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void doPost(HttpServletRequest request, HttpServletResponse resp)
|
||||
protected void doPost(HttpServletRequest request, HttpServletResponse response)
|
||||
throws ServletException, IOException {
|
||||
|
||||
PrintWriter writer = resp.getWriter();
|
||||
writer.print("<html>");
|
||||
writer.print("<head>");
|
||||
writer
|
||||
.print("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=UTF-8\" />");
|
||||
writer.print("<meta http-equiv=\"expires\" content=\"0\">");
|
||||
writer.print("<meta http-equiv=\"pragma\" content=\"no-cache\">");
|
||||
writer
|
||||
.print("<meta http-equiv=\"cache-control\" content=\"no-cache, must-revalidate\">");
|
||||
writer
|
||||
.print("<link rel=\"SHORTCUT ICON\" href=\"/plantuml/favicon.ico\">");
|
||||
writer.print("</head>");
|
||||
writer.print("<body>");
|
||||
writer
|
||||
.print("<h1>PlantUMLServer</h1><p>This application provides a servlet which serves images createdby <a href=\"http://plantuml.sourceforge.net\">PlantUML</a>.</p>");
|
||||
|
||||
String text = request.getParameter("text");
|
||||
String url = request.getParameter("url");
|
||||
String encode = "";
|
||||
String encoded = "";
|
||||
|
||||
Transcoder transcoder = getTranscoder();
|
||||
if (url != null) {
|
||||
// the URL form has been submitted
|
||||
if ((url != null) && (!url.trim().isEmpty())) {
|
||||
// TODO Verify the url is correct
|
||||
Pattern p = Pattern.compile(".*/(.*)");
|
||||
Matcher m = p.matcher(url);
|
||||
if (m.find()) {
|
||||
url = m.group(1);
|
||||
}
|
||||
text = transcoder.decode(url);
|
||||
}
|
||||
writer
|
||||
.print("<form method=post action=\"/plantuml/uml/post\"><textarea name=\"text\" cols=\"120\" rows=\"10\">");
|
||||
if (text != null) {
|
||||
encode = transcoder.encode(text);
|
||||
writer.print(text);
|
||||
}
|
||||
writer.print("</textarea><br><input type=\"submit\"></form>");
|
||||
writer.print("<hr>");
|
||||
writer.print("You can enter here a previously generated URL:<p>");
|
||||
|
||||
String host = "http://" + request.getServerName() + ":"
|
||||
+ request.getServerPort();
|
||||
String total = host + "/plantuml/uml/image/" + encode;
|
||||
|
||||
writer
|
||||
.print("<form method=\"post\" action=\"/plantuml/uml/post\"><input name=\"url\" type=\"text\" size=\"150\" value=\""
|
||||
+ total + "\">");
|
||||
writer.print("<br><input type=\"submit\"></form>");
|
||||
|
||||
if (text != null) {
|
||||
writer.print("<hr>");
|
||||
writer.print("You can use the following URL:<p>");
|
||||
|
||||
String urlPart = "\"" + total + "\"";
|
||||
writer.print("<a href=" + urlPart + " >");
|
||||
writer.print("<code>");
|
||||
writer.print("<img src=" + urlPart + " >");
|
||||
writer.print("</code></a><p>");
|
||||
|
||||
writer.print("<a href=" + urlPart + " >");
|
||||
writer.print("<img src=\"/plantuml/uml/image/" + encode + "\" >");
|
||||
writer.print("</a>");
|
||||
// the Text form has been submitted
|
||||
if ((text != null) && (!text.trim().isEmpty())) {
|
||||
encoded = transcoder.encode(text);
|
||||
}
|
||||
writer.print("</body></html>");
|
||||
writer.flush();
|
||||
|
||||
request.setAttribute("net.sourceforge.plantuml.servlet.decoded", text);
|
||||
request.setAttribute("net.sourceforge.plantuml.servlet.encoded", encoded);
|
||||
|
||||
// forward to index.jsp
|
||||
RequestDispatcher dispatcher = request.getRequestDispatcher("/index.jsp");
|
||||
dispatcher.forward(request, response);
|
||||
}
|
||||
|
||||
private Transcoder getTranscoder() {
|
||||
@ -162,8 +137,7 @@ public class PlantUmlServlet extends HttpServlet {
|
||||
|
||||
private void handleImageProxy(HttpServletResponse response, String num,
|
||||
String source) throws IOException {
|
||||
String s = getContent(source);
|
||||
SourceStringReader reader = new SourceStringReader(s);
|
||||
SourceStringReader reader = new SourceStringReader( getContent(source));
|
||||
int n = num == null ? 0 : Integer.parseInt(num);
|
||||
// Write the first image to "os"
|
||||
reader.generateImage(response.getOutputStream(), n);
|
||||
@ -175,16 +149,18 @@ public class PlantUmlServlet extends HttpServlet {
|
||||
plantUmlSource.append("@startuml\n");
|
||||
plantUmlSource.append(text);
|
||||
plantUmlSource.append("\n@enduml");
|
||||
|
||||
SourceStringReader reader = new SourceStringReader(plantUmlSource
|
||||
.toString());
|
||||
final String uml = plantUmlSource.toString();
|
||||
SourceStringReader reader = new SourceStringReader(uml);
|
||||
// Write the first image to "os"
|
||||
long today = System.currentTimeMillis();
|
||||
if ( StringUtils.isDiagramCacheable( uml)) {
|
||||
// Add http headers to force the browser to cache the image
|
||||
response.addDateHeader("Expires", today + 31536000000L);
|
||||
// today + 1 year
|
||||
response.addDateHeader("Last-Modified", 1261440000000L);
|
||||
// 2009 dec 22 constant date in the past
|
||||
response.addHeader("Cache-Control", "public");
|
||||
}
|
||||
response.setContentType("image/png");
|
||||
reader.generateImage(response.getOutputStream());
|
||||
|
||||
|
@ -5,11 +5,36 @@
|
||||
<servlet-name>plantumlservlet</servlet-name>
|
||||
<servlet-class>net.sourceforge.plantuml.servlet.PlantUmlServlet</servlet-class>
|
||||
</servlet>
|
||||
<!-- Patterns of the servlet -->
|
||||
<servlet-mapping>
|
||||
<servlet-name>plantumlservlet</servlet-name>
|
||||
<url-pattern>/uml/*</url-pattern>
|
||||
</servlet-mapping>
|
||||
<servlet-mapping>
|
||||
<servlet-name>plantumlservlet</servlet-name>
|
||||
<url-pattern>/form</url-pattern>
|
||||
</servlet-mapping>
|
||||
<servlet-mapping>
|
||||
<servlet-name>plantumlservlet</servlet-name>
|
||||
<url-pattern>/img/*</url-pattern>
|
||||
</servlet-mapping>
|
||||
<servlet-mapping>
|
||||
<servlet-name>plantumlservlet</servlet-name>
|
||||
<url-pattern>/start/*</url-pattern>
|
||||
</servlet-mapping>
|
||||
<servlet-mapping>
|
||||
<servlet-name>plantumlservlet</servlet-name>
|
||||
<url-pattern>/proxy/*</url-pattern>
|
||||
</servlet-mapping>
|
||||
<welcome-file-list>
|
||||
<welcome-file>uml</welcome-file>
|
||||
<welcome-file>/index.jsp</welcome-file>
|
||||
</welcome-file-list>
|
||||
<error-page>
|
||||
<exception-type>java.lang.Throwable</exception-type>
|
||||
<location>/error.jsp</location>
|
||||
</error-page>
|
||||
<error-page>
|
||||
<error-code>500</error-code>
|
||||
<location>/error.jsp</location>
|
||||
</error-page>
|
||||
</web-app>
|
||||
|
20
build.xml
20
build.xml
@ -1,13 +1,17 @@
|
||||
<project default="main" basedir=".">
|
||||
|
||||
<property name="tomcat-home" value="/tomcat" />
|
||||
<target name="main" depends="clean,compile,war">
|
||||
</target>
|
||||
|
||||
<target name="init">
|
||||
<!-- overwrite with your own values in a user.property file -->
|
||||
<property file="user.property" />
|
||||
<property name="tomcat-home" value="/tomcat" />
|
||||
<property name="debug" value="false" />
|
||||
<path id="project-classpath">
|
||||
<fileset dir="WEB-INF/lib" includes="*.jar" />
|
||||
<fileset dir="${tomcat-home}/lib" includes="*.jar" />
|
||||
</path>
|
||||
|
||||
<target name="main" depends="clean,compile,war">
|
||||
</target>
|
||||
|
||||
<target name="clean">
|
||||
@ -16,17 +20,21 @@
|
||||
<mkdir dir="WEB-INF/classes" />
|
||||
</target>
|
||||
|
||||
<target name="compile">
|
||||
<javac srcdir="WEB-INF/src" destdir="WEB-INF/classes" classpathref="project-classpath" />
|
||||
<target name="compile" depends="init">
|
||||
<javac srcdir="WEB-INF/src" destdir="WEB-INF/classes" debug="${debug}" classpathref="project-classpath" />
|
||||
</target>
|
||||
|
||||
<target name="war">
|
||||
<war destfile="plantuml.war"
|
||||
webxml="WEB-INF/web.xml">
|
||||
<classes dir="WEB-INF/classes" />
|
||||
<fileset file="favicon.ico" />
|
||||
<fileset dir="content" />
|
||||
<lib dir="WEB-INF/lib" />
|
||||
</war>
|
||||
</target>
|
||||
|
||||
<target name="deploy" depends="main">
|
||||
<copy file="plantuml.war" todir="${tomcat-home}/webapps" overwrite="true" />
|
||||
</target>
|
||||
|
||||
</project>
|
||||
|
32
content/error.jsp
Normal file
32
content/error.jsp
Normal file
@ -0,0 +1,32 @@
|
||||
<%@ page isErrorPage="true" contentType="text/html; charset=utf-8" pageEncoding="utf-8" %>
|
||||
<%
|
||||
String contextRoot = request.getContextPath();
|
||||
%>
|
||||
|
||||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
|
||||
<html xmlns="http://www.w3.org/1999/xhtml">
|
||||
<head>
|
||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
||||
<meta http-equiv="expires" content="0" />
|
||||
<meta http-equiv="pragma" content="no-cache" />
|
||||
<meta http-equiv="cache-control" content="no-cache, must-revalidate" />
|
||||
<link rel="stylesheet" href="<%=contextRoot %>/plantuml.css" type="text/css"/>
|
||||
<link rel="icon" href="<%=contextRoot %>/favicon.ico" type="image/x-icon"/>
|
||||
<link rel="shortcut icon" href="<%=contextRoot %>/favicon.ico" type="image/x-icon"/>
|
||||
<title>PlantUMLServer Error</title>
|
||||
</head>
|
||||
<body>
|
||||
<p>
|
||||
Sorry, but things didn't work out as planned.
|
||||
</p>
|
||||
<hr/>
|
||||
<jsp:useBean id="now" class="java.util.Date" />
|
||||
<ul>
|
||||
<li><%=now.toString() %></li>
|
||||
<li>Request that failed: <%=pageContext.getErrorData().getRequestURI() %></li>
|
||||
<li>Status code: <%=pageContext.getErrorData().getStatusCode() %></li>
|
||||
<li>Exception: <%=pageContext.getErrorData().getThrowable() %></li>
|
||||
</ul>
|
||||
<hr/>
|
||||
</body>
|
||||
</html>
|
Before Width: | Height: | Size: 1.1 KiB After Width: | Height: | Size: 1.1 KiB |
72
content/index.jsp
Normal file
72
content/index.jsp
Normal file
@ -0,0 +1,72 @@
|
||||
<%@ page info="index" contentType="text/html; charset=utf-8" pageEncoding="utf-8" %>
|
||||
|
||||
<%
|
||||
String contextRoot = request.getContextPath();
|
||||
String host = "http://" + request.getServerName() + ":" + request.getServerPort();
|
||||
String encoded = "";
|
||||
String umltext = "";
|
||||
String imgurl = "";
|
||||
Object encodedAttribute = request.getAttribute("net.sourceforge.plantuml.servlet.encoded");
|
||||
if (encodedAttribute != null) {
|
||||
encoded = encodedAttribute.toString();
|
||||
if (!encoded.isEmpty()) {
|
||||
imgurl = host + contextRoot + "/img/" + encoded;
|
||||
}
|
||||
}
|
||||
Object decodedAttribute = request.getAttribute("net.sourceforge.plantuml.servlet.decoded");
|
||||
if (decodedAttribute != null) {
|
||||
umltext = decodedAttribute.toString();
|
||||
}
|
||||
%>
|
||||
|
||||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
|
||||
<html xmlns="http://www.w3.org/1999/xhtml">
|
||||
<head>
|
||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
||||
<meta http-equiv="expires" content="0" />
|
||||
<meta http-equiv="pragma" content="no-cache" />
|
||||
<meta http-equiv="cache-control" content="no-cache, must-revalidate" />
|
||||
<link rel="stylesheet" href="<%=contextRoot %>/plantuml.css" type="text/css"/>
|
||||
<link rel="icon" href="<%=contextRoot %>/favicon.ico" type="image/x-icon"/>
|
||||
<link rel="shortcut icon" href="<%=contextRoot %>/favicon.ico" type="image/x-icon"/>
|
||||
<title>PlantUMLServer</title>
|
||||
</head>
|
||||
<body>
|
||||
<div id="header">
|
||||
<%-- PAGE TITLE --%>
|
||||
<h1>PlantUMLServer</h1>
|
||||
<p>This application provides a servlet which serves images created by <a href="http://plantuml.sourceforge.net">PlantUML</a>.</p>
|
||||
</div>
|
||||
<div id="content">
|
||||
<%-- CONTENT --%>
|
||||
<form method="post" action="<%=contextRoot %>/form">
|
||||
<p>
|
||||
<textarea name="text" cols="120" rows="10"><%=umltext %></textarea>
|
||||
<br/>
|
||||
<input type="submit" />
|
||||
</p>
|
||||
</form>
|
||||
<hr/>
|
||||
You can enter here a previously generated URL:
|
||||
<form method="post" action="<%=contextRoot %>/form">
|
||||
<p>
|
||||
<input name="url" type="text" size="150" value="<%=imgurl %>" />
|
||||
<br/>
|
||||
<input type="submit"/>
|
||||
</p>
|
||||
</form>
|
||||
<% if ( !imgurl.isEmpty()) { %>
|
||||
<hr/>
|
||||
<p>You can use the following URL:
|
||||
<br/>
|
||||
<a href="<%=imgurl %>"><code><img src="<%=imgurl %>" /></code></a>
|
||||
<br/><br/>
|
||||
<img id="diagram" src="<%=imgurl %>" alt="PlantUML diagram"/>
|
||||
</p>
|
||||
<% } //endif %>
|
||||
</div>
|
||||
|
||||
<%-- FOOTER
|
||||
<%@ include file="util/footer.jspf" %> --%>
|
||||
</body>
|
||||
</html>
|
20
content/plantuml.css
Normal file
20
content/plantuml.css
Normal file
@ -0,0 +1,20 @@
|
||||
/******************************
|
||||
* PlantUMLServlet style sheet *
|
||||
******************************/
|
||||
|
||||
/* Header */
|
||||
#header {
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
/* Form inputs */
|
||||
#content textarea, #content input[type=text] {
|
||||
width: 900px;
|
||||
}
|
||||
|
||||
/* Diagram */
|
||||
#content img#diagram {
|
||||
border: thin solid green;
|
||||
}
|
Loading…
Reference in New Issue
Block a user