/*
* Copyright 2007-2009 Hidekatsu Izuno
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
* either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/
package net.arnx.jsonic.web;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.Writer;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Date;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import net.arnx.jsonic.JSON;
import net.arnx.jsonic.JSONException;
import net.arnx.jsonic.JSONHint;
import static javax.servlet.http.HttpServletResponse.*;
import static net.arnx.jsonic.web.Container.*;
public class RESTServlet extends HttpServlet {
static final Map<String, String> DEFAULT_METHOD = new HashMap<String, String>();
static final Set<String> DEFAULT_VERB = new HashSet<String>();
static {
DEFAULT_METHOD.put("GET", "find");
DEFAULT_METHOD.put("POST", "create");
DEFAULT_METHOD.put("PUT", "update");
DEFAULT_METHOD.put("DELETE", "delete");
DEFAULT_VERB.add("HEAD");
DEFAULT_VERB.add("GET");
DEFAULT_VERB.add("POST");
DEFAULT_VERB.add("PUT");
DEFAULT_VERB.add("DELETE");
DEFAULT_VERB.add("OPTIONS");
}
static class Config {
public Class<? extends Container> container;
@JSONHint(anonym="target")
public Map<String, RouteMapping> mappings;
public Map<String, Pattern> definitions;
public Map<Class<? extends Exception>, Integer> errors;
public Map<String, String> method;
public Set<String> verb;
}
protected Container container;
Config config;
@Override
public void init(ServletConfig servletConfig) throws ServletException {
super.init(servletConfig);
String configText = servletConfig.getInitParameter("config");
JSON json = new JSON();
if (configText == null) {
Map<String, String> map = new HashMap<String, String>();
Enumeration<String> e = cast(servletConfig.getInitParameterNames());
while (e.hasMoreElements()) {
map.put(e.nextElement(), servletConfig.getInitParameter(e.nextElement()));
}
configText = json.format(map);
}
try {
config = json.parse(configText, Config.class);
if (config.container == null) config.container = Container.class;
container = json.parse(configText, config.container);
container.init(this);
} catch (Exception e) {
throw new ServletException(e);
}
if (config.definitions == null) config.definitions = new HashMap<String, Pattern>();
if (!config.definitions.containsKey("package")) config.definitions.put("package", Pattern.compile(".+"));
if (config.errors == null) config.errors = Collections.emptyMap();
if (config.method == null) config.method = DEFAULT_METHOD;
if (config.verb == null) config.verb = DEFAULT_VERB;
if (config.mappings == null) config.mappings = Collections.emptyMap();
for (Map.Entry<String, RouteMapping> entry : config.mappings.entrySet()) {
entry.getValue().init(entry.getKey(), config);
}
}
@Override
protected void doHead(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doREST(request, response);
}
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doREST(request, response);
}
@Override
protected void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doREST(request, response);
}
@Override
protected void doPut(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doREST(request, response);
}
@Override
protected void doDelete(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doREST(request, response);
}
@Override
protected void doOptions(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
doREST(request, response);
}
protected void doREST(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
String uri = (request.getContextPath().equals("/")) ?
request.getRequestURI() :
request.getRequestURI().substring(request.getContextPath().length());
File file = new File(getServletContext().getRealPath(uri));
if (file.exists()) {
OutputStream out = response.getOutputStream();
InputStream in = new FileInputStream(file);
try {
byte[] buffer = new byte[1024];
int count = 0;
while ((count = in.read(buffer)) > 0) {
out.write(buffer, 0, count);
}
} finally {
try {
if (in != null) in.close();
} catch (IOException e2) {
// no handle
}
}
return;
}
Route route = null;
for (RouteMapping m : config.mappings.values()) {
if ((route = m.matches(request, uri)) != null) {
container.debug("Route found: " + request.getMethod() + " " + uri);
break;
}
}
if (route == null) {
response.sendError(SC_NOT_FOUND, "Not Found");
return;
}
if (route.getHttpMethod() == null || route.getRestMethod() == null) {
container.debug("Method mapping not found: " + route.getHttpMethod());
response.sendError(SC_METHOD_NOT_ALLOWED, "Method Not Allowed");
return;
}
int status = SC_OK;
String callback = null;
Object result = null;
if ("GET".equals(request.getMethod())) {
callback = route.getParameter("callback");
} else if ("POST".equals(route.getHttpMethod())) {
status = SC_CREATED;
}
container.start(request, response);
JSON json = container.createJSON(request.getLocale());
try {
String className = route.getComponentClass(container);
Object component = container.getComponent(className);
if (component == null) {
throw new ClassNotFoundException("Component not found: " + className);
}
List<Object> params = null;
if (isJSONType(request.getContentType())) {
Object o = json.parse(request.getReader());
if (o instanceof List<?>) {
params = cast(o);
if (params.isEmpty()) {
params = new ArrayList<Object>(1);
params.add(route.getParameterMap());
} else if (params.get(0) instanceof Map<?, ?>) {
params.set(0, route.mergeParameterMap((Map<?, ?>)params.get(0)));
}
} else if (o instanceof Map<?, ?>) {
params = new ArrayList<Object>(1);
params.add(route.mergeParameterMap((Map<?, ?>)o));
} else {
throw new IllegalArgumentException("failed to convert parameters from JSON.");
}
} else {
params = new ArrayList<Object>(1);
params.add(route.getParameterMap());
}
Method method = container.getMethod(component, route.getRestMethod(), params);
if (method == null) {
throw new NoSuchMethodException("Method not found: " + route.getRestMethod());
}
json.setContext(component);
result = container.execute(json, component, method, params);
} catch (Exception e) {
if (e instanceof ClassNotFoundException) {
container.debug("Class Not Found.", e);
response.sendError(SC_NOT_FOUND, "Not Found");
response.flushBuffer();
} else if (e instanceof NoSuchMethodException) {
container.debug("Method Not Found.", e);
response.sendError(SC_NOT_FOUND, "Not Found");
response.flushBuffer();
} else if (e instanceof JSONException) {
container.debug("Fails to parse JSON.", e);
response.sendError(SC_BAD_REQUEST, "Bad Request");
response.flushBuffer();
} else if (e instanceof InvocationTargetException) {
Throwable cause = e.getCause();
container.debug("Cause error on invocation.", cause);
if (cause instanceof Error) {
throw (Error)cause;
} else if (cause instanceof IllegalStateException || cause instanceof UnsupportedOperationException) {
response.sendError(SC_NOT_FOUND, "Not Found");
response.flushBuffer();
} else if (cause instanceof IllegalArgumentException) {
response.sendError(SC_BAD_REQUEST, "Bad Request");
response.flushBuffer();
} else {
Integer errorCode = null;
for (Map.Entry<Class<? extends Exception>, Integer> entry : config.errors.entrySet()) {
if (entry.getKey().isAssignableFrom(cause.getClass()) && entry.getValue() != null) {
errorCode = entry.getValue();
break;
}
}
if (errorCode != null) {
response.setStatus(errorCode);
Map<String, Object> error = new LinkedHashMap<String, Object>();
error.put("name", cause.getClass().getSimpleName());
error.put("message", cause.getMessage());
error.put("data", cause);
result = error;
} else {
response.sendError(SC_INTERNAL_SERVER_ERROR, "Internal Server Error");
response.flushBuffer();
}
}
} else {
container.error("Internal error occurred.", e);
response.sendError(SC_INTERNAL_SERVER_ERROR, "Internal Server Error");
response.flushBuffer();
}
} finally {
container.end(request, response);
}
if (response.isCommitted()) return;
if (result == null
|| result instanceof CharSequence
|| result instanceof Boolean
|| result instanceof Number
|| result instanceof Date) {
if (status != SC_CREATED) status = SC_NO_CONTENT;
response.setStatus(status);
} else {
response.setContentType((callback != null) ? "text/javascript" : "application/json");
Writer writer = response.getWriter();
json.setPrettyPrint(container.isDebugMode());
if (callback != null) writer.append(callback).append("(");
json.format(result, writer);
if (callback != null) writer.append(");");
}
}
@Override
public void destroy() {
container.destory();
super.destroy();
}
static class RouteMapping {
static final Pattern PLACE_PATTERN = Pattern.compile("\\{\\s*(\\p{javaJavaIdentifierStart}[\\p{javaJavaIdentifierPart}\\.-]*)\\s*(?::\\s*((?:[^{}]|\\{[^{}]*\\})*)\\s*)?\\}");
static final Pattern DEFAULT_PATTERN = Pattern.compile("[^/().]+");
public String target;
public Map<String, String> method;
public Set<String> verb;
Config config;
Pattern pattern;
List<String> names;
public RouteMapping() {
}
public void init(String path, Config config) {
this.config = config;
this.names = new ArrayList<String>();
StringBuffer sb = new StringBuffer("^\\Q");
Matcher m = PLACE_PATTERN.matcher(path);
while (m.find()) {
String name = m.group(1);
names.add(name);
Pattern p = (m.group(2) != null) ? Pattern.compile(m.group(2)) : null;
if (p == null && config.definitions.containsKey(name)) {
p = config.definitions.get(name);
}
if (p == null) p = DEFAULT_PATTERN;
m.appendReplacement(sb, "\\\\E(" + p.pattern().replaceAll("\\((?!\\?)", "(?:").replace("\\", "\\\\") + ")\\\\Q");
}
m.appendTail(sb);
sb.append("\\E$");
this.pattern = Pattern.compile(sb.toString());
}
@SuppressWarnings({"unchecked", "rawtypes"})
public Route matches(HttpServletRequest request, String path) throws IOException {
Matcher m = pattern.matcher(path);
if (m.matches()) {
Map<String, Object> params = new HashMap<String, Object>();
for (int i = 0; i < names.size(); i++) {
String key = names.get(i);
Object value = m.group(i+1);
if (params.containsKey(key)) {
Object target = params.get(key);
if (target instanceof List) {
((List<Object>)target).add(value);
} else {
List<Object> list = new ArrayList<Object>(2);
list.add(target);
list.add(value);
}
} else {
params.put(key, value);
}
}
String httpMethod = request.getParameter("_method");
if (httpMethod == null) httpMethod = request.getMethod();
if (httpMethod != null) httpMethod = httpMethod.toUpperCase();
if (verb != null && !verb.contains(httpMethod)) {
httpMethod = null;
} else if (!config.verb.contains(httpMethod)) {
httpMethod = null;
}
Object restMethod = params.get("method");
if (restMethod instanceof List<?>) {
List<?> list = ((List<?>)restMethod);
restMethod = !list.isEmpty() ? list.get(0) : null;
}
if (restMethod == null && method != null) {
restMethod = method.get(httpMethod);
}
if (restMethod == null) {
restMethod = config.method.get(httpMethod);
}
parseParameter(request.getParameterMap(), (Map)params);
return new Route(httpMethod, (String)restMethod, target, params);
}
return null;
}
@SuppressWarnings("unchecked")
static void parseParameter(Map<String, String[]> pairs, Map<Object, Object> params) {
for (Map.Entry<String, String[]> entry : pairs.entrySet()) {
String name = entry.getKey();
boolean multiValue = false;
if (name.endsWith("[]")) {
name = name.substring(0, name.length()-2);
multiValue = true;
}
String[] values = entry.getValue();
int start = 0;
char old = '\0';
Map<Object, Object> current = params;
for (int i = 0; i < name.length(); i++) {
char c = name.charAt(i);
if (c == '.' || c == '[') {
String key = name.substring(start, (old == ']') ? i-1 : i);
Object target = current.get(key);
if (target instanceof Map) {
current = (Map<Object, Object>)target;
} else {
Map<Object, Object> map = new LinkedHashMap<Object, Object>();
if (target != null) map.put(null, target);
current.put(key, map);
current = map;
}
start = i+1;
}
old = c;
}
name = name.substring(start, (old == ']') ? name.length()-1 : name.length());
Object key = name;
if (name.length() > 0 && name.charAt(0) >= '0' && name.charAt(0) >= '9') {
try {
key = new BigDecimal(name);
} catch (Exception e) {
key = name;
}
}
if (current.containsKey(key)) {
Object target = current.get(key);
if (target instanceof Map) {
Map<Object, Object> map = (Map<Object, Object>)target;
if (map.containsKey(null)) {
target = map.get(null);
if (target instanceof List) {
List<Object> list = ((List<Object>)target);
for (String value : values) list.add(value);
} else {
List<Object> list = new ArrayList<Object>(values.length+1);
list.add(target);
for (String value : values) list.add(value);
map.put(null, list);
}
} else if (multiValue || values.length > 1) {
List<Object> list = new ArrayList<Object>(values.length);
for (String value : values) list.add(value);
map.put(null, list);
} else {
map.put(null, values[0]);
}
} else if (target instanceof List) {
List<Object> list = ((List<Object>)target);
for (String value : values) list.add(value);
} else {
List<Object> list = new ArrayList<Object>(values.length+1);
list.add(target);
for (String value : values) list.add(value);
current.put(key, list);
}
} else if (multiValue || values.length > 1) {
List<Object> list = new ArrayList<Object>(values.length);
for (String value : values) list.add(value);
current.put(key, list);
} else {
current.put(key, values[0]);
}
}
}
}
static class Route {
static final Pattern REPLACE_PATTERN = Pattern.compile("\\$\\{(\\p{javaJavaIdentifierStart}\\p{javaJavaIdentifierPart}*)\\}");
String target;
String httpMethod;
String restMethod;
Map<Object, Object> params;
public Route(String httpMethod, String restMethod, String target, Map<String, Object> params) throws IOException {
this.httpMethod = httpMethod;
this.restMethod = restMethod;
this.target = target;
this.params = cast(params);
}
public String getHttpMethod() {
return httpMethod;
}
public String getRestMethod() {
return restMethod;
}
public String getParameter(String name) {
Object o = params.get(name);
if (o instanceof Map<?, ?>) {
Map<?, ?> map = (Map<?, ?>)o;
if (map.containsKey(null)) o = map.get(null);
}
if (o instanceof List<?>) {
List<?> list = (List<?>)o;
if (!list.isEmpty()) o = list.get(0);
}
return (o instanceof String) ? (String)o : null;
}
public Map<?, ?> getParameterMap() {
return params;
}
public String getComponentClass(Container container) {
Matcher m = REPLACE_PATTERN.matcher(target);
StringBuffer sb = new StringBuffer();
while (m.find()) {
String key = m.group(1);
String value = getParameter(key);
if (key.equals("class") && container.namingConversion) {
value = toUpperCamel(value);
} else if (key.equals("package")) {
value = value.replace('/', '.');
}
m.appendReplacement(sb, (value != null) ? value : "");
}
m.appendTail(sb);
return sb.toString();
}
@SuppressWarnings("unchecked")
public Map<?, ?> mergeParameterMap(Map<?, ?> newParams) {
for (Map.Entry<?, ?> entry : newParams.entrySet()) {
if (params.containsKey(entry.getKey())) {
Object target = params.get(entry.getKey());
if (target instanceof Map) {
Map<Object, Object> map = (Map<Object, Object>)target;
if (map.containsKey(null)) {
target = map.get(null);
if (target instanceof List) {
((List<Object>)target).add(entry.getValue());
} else {
List<Object> list = new ArrayList<Object>();
list.add(target);
list.add(entry.getValue());
map.put(null, list);
}
} else {
map.put(null, entry.getValue());
}
} else if (target instanceof List) {
((List<Object>)target).add(entry.getValue());
} else {
List<Object> list = new ArrayList<Object>();
list.add(target);
list.add(entry.getValue());
params.put(entry.getKey(), list);
}
} else {
params.put(entry.getKey(), entry.getValue());
}
}
return params;
}
}
}