net.sf.aguacate.AguacateServlet Maven / Gradle / Ivy
package net.sf.aguacate;
import java.io.IOException;
import java.text.MessageFormat;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import net.sf.aguacate.configuration.Configuration;
import net.sf.aguacate.configuration.ConfigurationCoupling;
import net.sf.aguacate.context.ContextValidationResult;
import net.sf.aguacate.function.FunctionEvalResult;
import net.sf.aguacate.http.HttpBodyBuilerCoupling;
import net.sf.aguacate.http.HttpBodyParserCoupling;
import net.sf.aguacate.http.HttpMethods;
import net.sf.aguacate.util.meta.Meta;
import net.sf.aguacate.util.meta.MetaCoupling;
import net.sf.aguacate.util.servlet.ServletUtil;
import net.sf.aguacate.validator.InputValidationResponse;
import net.sf.aguacate.validator.ValidationConversionResult;
public class AguacateServlet extends HttpServlet {
private static final long serialVersionUID = 6335723473241984651L;
private static final Logger LOGGER = LogManager.getLogger(AguacateServlet.class);
private static final Logger LOGGER_INIT = LogManager.getLogger("aguacate.init");
private static final Logger LOGGER_REQUEST = LogManager.getLogger("aguacate.request");
@Override
public void init() throws ServletException {
super.init();
String group = "net.sf.aguacate";
String artifact = "aguacate-servlet";
Meta meta = MetaCoupling.get(AguacateServlet.class, group, artifact);
String message = group + " & " + artifact + " & " + meta.getVersion();
log(message);
LOGGER_INIT.info(message);
}
@Override
protected void service(HttpServletRequest request, HttpServletResponse response) {
if (LOGGER_REQUEST.isInfoEnabled()) {
String servletPath = request.getServletPath();
String pathInfo = request.getPathInfo();
LOGGER_REQUEST.info("request for: {} ~ {}", servletPath, pathInfo);
}
String entity = ServletUtil.getEntity(request);
String method = request.getMethod();
try {
Configuration configuration = ConfigurationCoupling.load(entity);
if (configuration == null) {
response.setStatus(HttpServletResponse.SC_NOT_FOUND);
} else {
if (HttpMethods.acceptsPathParameters(method)) {
String id = ServletUtil.getId(request);
if (id == null) {
if (HttpMethods.optionalPathParameters(method)) {
String method0 = method + '0';
if (configuration.accepts(method0)) {
if (HttpMethods.acceptsBodyRequest(method0)) {
String contentType = request.getContentType();
if (contentType == null || contentType.isEmpty()) {
response.setStatus(HttpServletResponse.SC_NOT_ACCEPTABLE);
} else {
Map body = HttpBodyParserCoupling.defaultInstance()
.parse(request);
if (body == null) {
response.setStatus(HttpServletResponse.SC_NOT_ACCEPTABLE);
} else {
validate(configuration, method0, request, response, null, body);
}
}
} else {
validate(configuration, method0, request, response);
}
} else {
if ("OPTIONS".equals(method)) {
// TODO: Allow some kind of cache for this string
response.setHeader("Allow", toHeaderValue(configuration));
response.setStatus(HttpServletResponse.SC_NO_CONTENT);
} else {
response.setStatus(HttpServletResponse.SC_METHOD_NOT_ALLOWED);
}
}
} else {
// Invalid state
response.setStatus(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
}
} else {
if (configuration.accepts(method)) {
if (HttpMethods.acceptsBodyRequest(method)) {
String contentType = request.getContentType();
if (contentType == null || contentType.isEmpty()) {
response.setStatus(HttpServletResponse.SC_NOT_ACCEPTABLE);
} else {
Map body = HttpBodyParserCoupling.defaultInstance().parse(request);
if (body == null) {
response.setStatus(HttpServletResponse.SC_NOT_ACCEPTABLE);
} else {
validate(configuration, method, request, response, id, body);
}
}
} else {
validate(configuration, method, request, response, id);
}
} else {
if ("OPTIONS".equals(method)) {
// TODO: Allow some kind of cache for this string
response.setHeader("Allow", toHeaderValue(configuration));
response.setStatus(HttpServletResponse.SC_NO_CONTENT);
} else {
response.setStatus(HttpServletResponse.SC_METHOD_NOT_ALLOWED);
}
}
}
} else {
if (configuration.accepts(method)) {
if (HttpMethods.acceptsBodyRequest(method)) {
String contentType = request.getContentType();
if (contentType == null || contentType.isEmpty()) {
response.setStatus(HttpServletResponse.SC_NOT_ACCEPTABLE);
} else {
Map body = HttpBodyParserCoupling.defaultInstance().parse(request);
if (body == null) {
response.setStatus(HttpServletResponse.SC_NOT_ACCEPTABLE);
} else {
validate(configuration, method, request, response, body);
}
}
} else {
validate(configuration, method, request, response);
}
} else {
if ("OPTIONS".equals(method)) {
// TODO: Allow some kind of cache for this string
response.setHeader("Allow", toHeaderValue(configuration));
response.setStatus(HttpServletResponse.SC_NO_CONTENT);
} else {
response.setStatus(HttpServletResponse.SC_METHOD_NOT_ALLOWED);
}
}
}
}
} catch (RuntimeException | IOException e) {
LOGGER.error(MessageFormat.format("{0}, {1}", entity, method), e);
response.setStatus(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
}
}
String toHeaderValue(Configuration configuration) {
String[] allowed = configuration.methods();
StringBuilder builder = new StringBuilder("OPTIONS");
boolean get = false;
for (String current : allowed) {
if ("GET0".equals(current)) {
if (!get) {
get = true;
builder.append(", ");
builder.append("GET");
}
} else {
if ("GET".equals(current)) {
if (!get) {
builder.append(", ");
builder.append(current);
}
} else {
builder.append(", ");
builder.append(current);
}
}
}
String value = builder.toString();
return value;
}
// GET
void validate(Configuration configuration, String method, HttpServletRequest request, HttpServletResponse response)
throws IOException {
LOGGER.info("GET0: {}", method);
processInput(configuration, method, new InputValidationResponse(new HashMap<>(), null), response);
}
// GET, DELETE
void validate(Configuration configuration, String method, HttpServletRequest request, HttpServletResponse response,
String id) throws IOException {
LOGGER.info("GET, DELETE: {}", method);
processInput(configuration, method,
configuration.getValidatorConverter(method).validate(id, Collections.emptyMap()), response);
}
// PUT, PATCH
void validate(Configuration configuration, String method, HttpServletRequest request, HttpServletResponse response,
String id, Map body) throws IOException {
LOGGER.info("PUT, DELETE: {}, {}", method, id);
processInput(configuration, method, configuration.getValidatorConverter(method).validate(id, body), response);
}
// POST
void validate(Configuration configuration, String method, HttpServletRequest request, HttpServletResponse response,
Map body) throws IOException {
LOGGER.info("POST: {}", method);
processInput(configuration, method, configuration.getValidatorConverter(method).validate(body), response);
}
// TODO: move this method
void processInput(Configuration configuration, String method, InputValidationResponse validationResponse,
HttpServletResponse response) throws IOException {
if (validationResponse.isSuccess()) {
Map context = validationResponse.getContext();
LOGGER.trace("initial context: {}", context);
ContextValidationResult result = configuration.getContextValidator().validate(method, context);
if (result.isSuccess()) {
configuration.getContextProcessor().process(response, method, context, configuration.getOutputFields(method));
} else {
response.setStatus(HttpServletResponse.SC_BAD_REQUEST);
Map map = new HashMap<>();
map.put("type", "RULE");
List rules = new ArrayList<>();
List errors = result.getErrors();
LOGGER.trace("errors: {}", errors);
for (FunctionEvalResult error : errors) {
rules.add(error.getMessage());
}
map.put("rules", rules);
HttpBodyBuilerCoupling.defaultInstance().build(response, map);
}
} else {
response.setStatus(HttpServletResponse.SC_BAD_REQUEST);
Map map = new HashMap<>();
map.put("type", "VALIDATION");
Map validations = new HashMap<>();
Map failures = validationResponse.getFailures();
LOGGER.trace("failures: {}", failures);
for (Map.Entry entry : failures.entrySet()) {
validations.put(entry.getKey(), entry.getValue().getMessage());
}
map.put("validations", validations);
HttpBodyBuilerCoupling.defaultInstance().build(response, map);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy