com.idrsolutions.microservice.BaseServlet Maven / Gradle / Ivy
Show all versions of base-microservice-example Show documentation
/*
* Base Microservice Example
*
* Project Info: https://github.com/idrsolutions/base-microservice-example
*
* Copyright 2021 IDRsolutions
*
* 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 com.idrsolutions.microservice;
import com.idrsolutions.microservice.utils.DownloadHelper;
import com.idrsolutions.microservice.utils.FileHelper;
import com.idrsolutions.microservice.utils.HttpHelper;
import javax.json.Json;
import javax.json.stream.JsonParser;
import javax.json.stream.JsonParsingException;
import javax.naming.SizeLimitExceededException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.Part;
import java.io.*;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.ScheduledExecutorService;
import java.util.logging.Level;
import java.util.logging.Logger;
/**
* An extendable base for conversion microservices. Provides general
* functionality for polling, file upload/download, initial creation of files
* and UUID's.
*/
public abstract class BaseServlet extends HttpServlet {
private static final Logger LOG = Logger.getLogger(BaseServlet.class.getName());
protected static final String USER_HOME;
static {
String userDir = System.getProperty("user.home");
if (!userDir.endsWith("/") && !userDir.endsWith("\\")) {
userDir += System.getProperty("file.separator");
}
USER_HOME = userDir;
}
private static String INPUTPATH = USER_HOME + ".idr/input/";
private static String OUTPUTPATH = USER_HOME + ".idr/output/";
private static long individualTTL = 86400000L; // 24 hours
private static final int NUM_DOWNLOAD_RETRIES = 2;
private final ConcurrentHashMap imap = new ConcurrentHashMap<>();
/**
* Get the location where input files is stored
* @return inputPath the path where input files is stored
*/
public static String getInputPath() {
return INPUTPATH;
}
/**
* Get the location where the converter output is stored
*
* @return outputPath the path where output files is stored
*/
public static String getOutputPath() {
return OUTPUTPATH;
}
/**
* Get the time to live of individuals on the server (The duration that the
* information of an individual is kept on the server)
*
* @return individualTTL the time to live of an individual
*/
public static long getIndividualTTL() {
return individualTTL;
}
/**
* Set the location where input files is stored
*
* @param inputPath the path where input files is stored
*/
public static void setInputPath(final String inputPath) {
INPUTPATH = inputPath;
}
/**
* Set the location where the converter output is stored
*
* @param outputPath the path where output files is stored
*/
public static void setOutputPath(final String outputPath) {
OUTPUTPATH = outputPath;
}
/**
* Set the time to live of individuals on the server (The duration that the
* information of an individual is kept on the server)
*
* @param ttlDuration the time to live of an individual
*/
public static void setIndividualTTL(final long ttlDuration) {
individualTTL = ttlDuration;
}
/**
* Set an HTTP error code and message to the given response.
*
* @param request the HttpServletRequest object to reply to
* @param response the HttpServletResponse object on which to send the
* response
* @param error the error message to pass in the body of the client
* @param status the HTTP status to set the response to response.
*/
protected static void doError(final HttpServletRequest request, final HttpServletResponse response, final String error, final int status) {
response.setStatus(status);
sendResponse(request, response, Json.createObjectBuilder().add("error", error).build().toString());
}
/**
* Send a JSON response
*
* @param request the HttpServletRequest object to reply to
* @param response the HttpServletResponse object on which to send the
* response
* @param content the JSON response to send
*/
private static void sendResponse(final HttpServletRequest request, final HttpServletResponse response, final String content) {
allowCrossOrigin(request, response);
response.setContentType("application/json");
try (final PrintWriter out = response.getWriter()) {
out.println(content);
} catch (final IOException e) {
LOG.log(Level.SEVERE, "IOException thrown when sending json response", e);
}
}
/**
* Get request to the servlet. See API docs in respective end servlets for
* more information.
*
* @param request the request from the client
* @param response the response to send once this method exits
* @see Individual#toJsonString()
*/
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) {
final String uuidStr = request.getParameter("uuid");
if (uuidStr == null) {
doError(request, response, "No uuid provided", 404);
return;
}
final Individual individual = imap.get(uuidStr);
if (individual == null) {
doError(request, response, "Unknown uuid: " + uuidStr, 404);
return;
}
sendResponse(request, response, individual.toJsonString());
}
/**
* Writes to response object with the communication methods that this server
* supports.
*
* @param request the request from the client
* @param response the response to send once this method exits
* @see BaseServlet#allowCrossOrigin(HttpServletRequest, HttpServletResponse)
*/
@Override
protected void doOptions(HttpServletRequest request, HttpServletResponse response) {
allowCrossOrigin(request, response);
}
/**
* Allow cross origin requests according to the CORS standard.
* This method will not override any existing headers that are already set in the response.
*
* @param request the request from the client
* @param response the response object to the request from the client
*/
private static void allowCrossOrigin(final HttpServletRequest request, final HttpServletResponse response) {
final String credentials = response.getHeader("Access-Control-Allow-Credentials");
if (credentials == null) {
response.addHeader("Access-Control-Allow-Credentials", "true");
}
final String origin = response.getHeader("Access-Control-Allow-Origin");
if (origin == null) {
String requestOrigin = request.getHeader("origin");
if (requestOrigin == null) {
requestOrigin = "*";
}
response.addHeader("Access-Control-Allow-Origin", requestOrigin);
}
final String methods = response.getHeader("Access-Control-Allow-Methods");
if (methods == null) {
response.addHeader("Access-Control-Allow-Methods", "GET, PUT, POST, OPTIONS, DELETE");
}
final String headers = response.getHeader("Access-Control-Allow-Headers");
if (headers == null) {
response.addHeader("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Access-Control-Allow-Origin, authorization");
}
}
/**
* A post request to the server.
*
* @param request the request from the client
* @param response the response to send once this method exits
* @see BaseServlet#convert(Individual, Map, File, File, String)
*/
@Override
protected void doPost(final HttpServletRequest request, final HttpServletResponse response) {
imap.entrySet().removeIf(entry -> entry.getValue().getTimestamp() < new Date().getTime() - individualTTL);
final String inputType = request.getParameter("input");
if (inputType == null) {
doError(request, response, "Missing input type", 400);
return;
}
final String uuidStr = UUID.randomUUID().toString();
final Individual individual = new Individual(uuidStr);
if (!validateRequest(request, response, individual)) {
return;
}
individual.setCustomData(request.getAttribute("com.idrsolutions.microservice.customData"));
final Map parameterMap = new HashMap<>(request.getParameterMap());
switch (inputType) {
case "upload":
if (!handleFileFromRequest(individual, request, response, parameterMap)) {
return;
}
break;
case "download":
if (!handleFileFromUrl(individual, request, response, parameterMap)) {
return;
}
break;
default:
doError(request, response, "Unrecognised input type", 400);
return;
}
imap.put(uuidStr, individual);
sendResponse(request, response, Json.createObjectBuilder().add("uuid", uuidStr).build().toString());
}
/**
* Sanitize the file name by removing all none url/filepath friendly
* characters.
*
* @param fileName the filename to sanitize
* @return the sanitized filename
*/
private static String sanitizeFileName(final String fileName) {
final int extPos = fileName.lastIndexOf('.');
// Limit filenames to chars allowed in unencoded URLs and Windows filenames for now
final String fileNameWithoutExt = fileName.substring(0, extPos).replaceAll("[^$\\-_.+!'(),a-zA-Z0-9]", "_");
final String ext = fileName.substring(extPos + 1);
return fileNameWithoutExt + '.' + ext;
}
/**
* Create the input directory for the clients file.
*
* @param uuid the uuid to use to create the directory
* @return the input directory
*/
private static File createInputDirectory(final String uuid) {
final String userInputDirPath = INPUTPATH + uuid;
final File inputDir = new File(userInputDirPath);
if (!inputDir.exists()) {
inputDir.mkdirs();
}
return inputDir;
}
/**
* Create the output directory to store the converted pdf at.
*
* @param uuid the uuid to use to create the output directory
* @return the output directory
*/
private static File createOutputDirectory(final String uuid) {
final String userOutputDirPath = OUTPUTPATH + uuid;
final File outputDir = new File(userOutputDirPath);
if (outputDir.exists()) {
FileHelper.deleteFolder(outputDir);
}
outputDir.mkdirs();
return outputDir;
}
/**
* Handle and convert file uploaded in the request.
*
* This method blocks until the file is initially processed and exists when
* the conversion begins.
*
* @param individual the individual associated with this conversion
* @param request the request for this conversion
* @param response the response object for the request
* @param params the parameter map from the request
* @return true on success, false on failure
*/
private boolean handleFileFromRequest(final Individual individual, final HttpServletRequest request,
final HttpServletResponse response, final Map params) {
final Part filePart;
try {
filePart = request.getPart("file");
} catch (IOException e) {
LOG.log(Level.SEVERE, "IOException when getting the file part", e);
doError(request, response, "Error handling file", 500);
return false;
} catch (ServletException e) {
doError(request, response, "Missing file", 400);
return false;
}
if (filePart == null) {
doError(request, response, "Missing file", 400);
return false;
}
final long fileSizeLimit = getFileSizeLimit(request);
if (fileSizeLimit > 0 && filePart.getSize() > fileSizeLimit) {
doError(request, response, "File size limit exceeded", 400);
return false;
}
final String originalFileName = getFileName(filePart);
if (originalFileName == null) {
doError(request, response, "Missing file name", 400);
return false;
}
if (originalFileName.indexOf('.') == -1) {
doError(request, response, "File has no extension", 400);
return false;
}
final File inputFile;
try {
final InputStream fileContent = filePart.getInputStream();
final byte[] fileBytes = new byte[(int) filePart.getSize()];
fileContent.read(fileBytes);
fileContent.close();
inputFile = outputFile(originalFileName, individual, fileBytes);
} catch (final IOException e) {
LOG.log(Level.SEVERE, "IOException when reading an uploaded file", e);
doError(request, response, "Internal error", 500); // Failed to save file to disk
return false;
}
final File outputDir = createOutputDirectory(individual.getUuid());
addToQueue(individual, params, inputFile, outputDir, getContextURL(request));
return true;
}
/**
* Handle and convert a file located at a given url.
*
* This method does not block when attempting to download the file from the
* url.
*
* @param individual the individual associated with this conversion
* @param request the request for this conversion
* @param response the response object for the request
* @param params the parameter map from the request
* @return true on initial success (url has been provided)
*/
private boolean handleFileFromUrl(final Individual individual, final HttpServletRequest request,
final HttpServletResponse response, final Map params) {
String url = request.getParameter("url");
if (url == null || url.isEmpty()) {
doError(request, response, "No url given", 400);
return false;
}
if (!url.toLowerCase().startsWith("http://") && !url.toLowerCase().startsWith("https://")) {
doError(request, response, "Unsupported protocol", 400);
return false;
}
// This does not need to be asynchronous
String filename = DownloadHelper.getFileNameFromUrl(url);
// In case a filename cannot be parsed from the url.
if (filename == null) {
filename = "document.pdf";
}
if (filename.indexOf('.') == -1) {
doError(request, response, "File has no extension", 400);
return false;
}
final long fileSizeLimit = getFileSizeLimit(request);
if (fileSizeLimit > 0) {
long fileSize;
try {
fileSize = DownloadHelper.getFileSizeFromUrl(url);
} catch (IOException e) {
LOG.log(Level.SEVERE, "IOException when finding the FileSize of a remote file", e);
doError(request, response, "Internal error", 500);
return false;
}
if (fileSize > fileSizeLimit) {
doError(request, response, "File size limit exceeded", 400);
return false;
}
}
// To allow use in lambda function.
final String finalFilename = filename;
final String contextUrl = getContextURL(request);
final ExecutorService downloadQueue = (ExecutorService) getServletContext().getAttribute("downloadQueue");
downloadQueue.submit(() -> {
File inputFile = null;
try {
final byte[] fileBytes = DownloadHelper.getFileFromUrl(url, NUM_DOWNLOAD_RETRIES, fileSizeLimit);
inputFile = outputFile(finalFilename, individual, fileBytes);
} catch (IOException e) {
individual.doError(1200, "Could not get file from URL");
} catch (SizeLimitExceededException e) {
individual.doError(1210, "File exceeds file size limit");
}
final File outputDir = createOutputDirectory(individual.getUuid());
addToQueue(individual, params, inputFile, outputDir, contextUrl);
});
return true;
}
/**
* Add a conversion task to the thread queue.
*
* @param individual the individual belonging to this conversion
* @param params the parameter map from the request
* @param inputFile the input file to convert
* @param outputDir the output directory to convert to
* @param contextUrl the context url of the servlet
*/
private void addToQueue(final Individual individual, final Map params, final File inputFile,
final File outputDir, final String contextUrl) {
final ExecutorService convertQueue = (ExecutorService) getServletContext().getAttribute("convertQueue");
convertQueue.submit(() -> {
try {
convert(individual, params, inputFile, outputDir, contextUrl);
} finally {
handleCallback(individual, params);
individual.setAlive(false);
}
});
}
/**
* Validate the request to ensure suitable for the microservice conversion,
* failure will lead to the request stopping before starting the conversion.
*
* It is recommended to call doError and set the individual conversionParams
* from inside implementations of validateRequest.
*
* @param request the request for this conversion
* @param response the response object for the request
* @param individual the individual belonging to this conversion
* @return true if the request is valid, false if not
*/
protected abstract boolean validateRequest(final HttpServletRequest request, final HttpServletResponse response,
final Individual individual);
/**
* This method converts a file and writes it to the output directory under
* the Individual's UUID.
*
* @param individual Internal representation of individual who made this
* request
* @param params the map of parameters from the request
* @param inputFile the File to convert
* @param outputDir the directory the converted file should be written to
* @param contextUrl The url from the protocol up to the servlet url
* pattern.
*/
protected abstract void convert(Individual individual, Map params,
File inputFile, File outputDir, String contextUrl);
/**
* Write the given file bytes to the output directory under filename.
*
* @param filename the filename to output to
* @param individual the individual that began the conversion request
* @param fileBytes the bytes to be written.
* @return the created file
* @throws IOException on file not being writable
*/
private File outputFile(String filename, Individual individual, byte[] fileBytes) throws IOException {
final File inputDir = createInputDirectory(individual.getUuid());
final File inputFile = new File(inputDir, sanitizeFileName(filename));
try (final FileOutputStream output = new FileOutputStream(inputFile)) {
output.write(fileBytes);
output.flush();
}
return inputFile;
}
/**
* Get the filename of the file contained in this request part.
*
* @param part the file part from the HTTP request
* @return the file name or null if it does not exist
*/
private String getFileName(final Part part) {
for (String content : part.getHeader("content-disposition").split(";")) {
if (content.trim().startsWith("filename")) {
return content.substring(
content.indexOf('=') + 1).trim().replace("\"", "");
}
}
return null;
}
/**
* Checks if the callbackUrl parameter was included in the request, if so it
* will queue the callback into the callbackQueue.
*
* @param individual the Individual that is sent to the URL
* @param params the request parameters
*/
private void handleCallback(final Individual individual, final Map params) {
final String[] rawParam = params.get("callbackUrl");
if (rawParam != null && rawParam.length > 0) {
final String callbackUrl = rawParam[0];
if (!callbackUrl.equals("")) {
final ScheduledExecutorService callbackQueue = (ScheduledExecutorService) getServletContext().getAttribute("callbackQueue");
callbackQueue.submit(() -> HttpHelper.sendCallback(callbackUrl, individual.toJsonString(), callbackQueue, 1));
}
}
}
/**
* Gets the full URL before the part containing the path(s) specified in
* urlPatterns of the servlet.
*
* @param request the request from the client
* @return protocol://servername/contextPath
*/
protected static String getContextURL(final HttpServletRequest request) {
final StringBuffer full = request.getRequestURL();
return full.substring(0, full.length() - request.getServletPath().length());
}
/**
* Try to get the fileSizeLimit attribute as a long from the
* HttpServeletRequest
*
* @param request the request from the client
* @return the value of fileSizeLimit or -1 if the attribute is not set
*/
private static long getFileSizeLimit(final HttpServletRequest request) {
final Object rawSizeLimit = request.getAttribute("com.idrsolutions.microservice.fileSizeLimit");
if (rawSizeLimit != null && rawSizeLimit instanceof Long) {
return (long) rawSizeLimit;
} else {
return -1L;
}
}
/**
* Get the conversion parameters from a JSON string.
*
* JSON Array values are ignored.
* Embedded objects have all k/v extracted (but the key for the object is lost).
*
* @param settings a JSON string of settings
* @return a Map made from the JSON k/v
* @throws JsonParsingException on issue with JSON parsing
*/
protected static Map parseSettings(final String settings) throws JsonParsingException {
final Map out = new HashMap<>();
if (settings == null || settings.isEmpty()) {
return out;
}
try (final JsonParser jp = Json.createParser(new StringReader(settings))) {
String currentKey = null;
byte arrayDepth = 0;
while (jp.hasNext()) {
JsonParser.Event e = jp.next();
switch (e) {
case START_ARRAY:
arrayDepth++;
break;
case END_ARRAY:
arrayDepth--;
break;
case KEY_NAME:
currentKey = jp.getString();
break;
case VALUE_STRING:
if (currentKey != null && arrayDepth < 1) {
out.put(currentKey, jp.getString());
}
break;
case VALUE_NUMBER:
if (currentKey != null && arrayDepth < 1) {
out.put(currentKey, String.valueOf(jp.getInt()));
}
break;
case VALUE_TRUE:
if (currentKey != null && arrayDepth < 1) {
out.put(currentKey, "true");
}
break;
case VALUE_FALSE:
if (currentKey != null && arrayDepth < 1) {
out.put(currentKey, "false");
}
break;
}
}
}
return out;
}
}