All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.dropwizard.bundles.webjars.WebJarServlet Maven / Gradle / Ivy

package io.dropwizard.bundles.webjars;

import com.google.common.base.Splitter;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.LoadingCache;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
import com.google.common.net.HttpHeaders;
import java.io.IOException;
import java.util.concurrent.TimeUnit;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.ws.rs.core.EntityTag;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * A servlet that will load resources from WebJars found in the classpath.  In order to make it more
 * convenient to use a WebJar, this servlet will automatically determine the version of the WebJar
 * present in the classpath and make it so that you don't need to explicitly specify a version
 * number as part of the URL. This allows WebJars to be upgraded entirely via maven dependencies
 * without having to update all of the references to the WebJar in your UI code.
 */
public class WebJarServlet extends HttpServlet {
  /** The default URL prefix that webjars are served out of. */
  public static final String DEFAULT_URL_PREFIX = "/webjars/";

  /** The default maven group(s) that WebJars are searched for in. */
  public static final String[] DEFAULT_MAVEN_GROUPS = {"org.webjars"};

  /** An If-None-Match header parser, splits the header into the multiple ETags if present. */
  private static final Splitter IF_NONE_MATCH_SPLITTER =
      Splitter.on(',').omitEmptyStrings().trimResults();

  private static final Logger LOG = LoggerFactory.getLogger(WebJarServlet.class);

  /** The URL prefix that webjars are served out of. */
  private final String urlPrefix;

  /** A path parser that can determine the library and library resource a particular path is for. */
  private final Pattern pathParser;

  private final transient LoadingCache cache;

  /**
   * Construct the actual servlet that this bundle will install.
   *
   * @param builder The cache definition for our webjar bundle.
   * @param groups The allowed maven groups of webjars to look for and match against.
   */
  @SuppressWarnings("unchecked")
  public WebJarServlet(CacheBuilder builder, Iterable groups, String urlPrefix) {
    if (builder == null) {
      builder = CacheBuilder.newBuilder()
          .maximumWeight(5 * 1024 * 1024)
          .expireAfterAccess(5, TimeUnit.MINUTES);
    }

    if (groups == null || Iterables.isEmpty(groups)) {
      groups = ImmutableList.copyOf(DEFAULT_MAVEN_GROUPS);
    }

    AssetLoader loader = new AssetLoader(new VersionLoader(groups));
    cache = builder.weigher(new AssetWeigher()).build(loader);
    this.urlPrefix = urlPrefix;
    pathParser = Pattern.compile(urlPrefix + "([^/]+)/(.+)");
  }

  @Override
  protected void doGet(HttpServletRequest req, HttpServletResponse resp)
      throws ServletException, IOException {
    try {
      handle(req, resp);
    } catch (Exception e) {
      LOG.info("Error processing request: {}", req, e);
      resp.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR);
    }
  }

  private void handle(HttpServletRequest req, HttpServletResponse resp) throws IOException {
    String path = getFullPath(req);

    // Check to see if this is a valid path that we know how to deal with.
    // If so parse out the library and resource.
    Matcher match = pathParser.matcher(path);
    if (!match.matches()) {
      resp.sendError(HttpServletResponse.SC_NOT_FOUND);
      return;
    }

    // The path is valid, try to load the asset.
    AssetId id = new AssetId(match.group(1), match.group(2));
    Asset asset = cache.getUnchecked(id);
    if (asset == AssetLoader.NOT_FOUND) {
      resp.sendError(HttpServletResponse.SC_NOT_FOUND);
      return;
    }

    // We know we've found the asset. No matter what happens, make sure we send back its last
    // modification time as well as its ETag
    resp.setDateHeader(HttpHeaders.LAST_MODIFIED, asset.lastModifiedTime);
    resp.setHeader(HttpHeaders.ETAG, hash2etag(asset.hash));

    // Check the If-None-Match header to see if any ETags match this resource
    String ifNoneMatch = req.getHeader(HttpHeaders.IF_NONE_MATCH);
    if (ifNoneMatch != null) {
      for (String etag : IF_NONE_MATCH_SPLITTER.split(ifNoneMatch)) {
        if ("*".equals(etag) || asset.hash.equals(etag2hash(etag))) {
          resp.sendError(HttpServletResponse.SC_NOT_MODIFIED);
          return;
        }
      }
    }

    // Check the If-Modified-Since header to see if this resource is newer
    if (asset.lastModifiedTime <= req.getDateHeader(HttpHeaders.IF_MODIFIED_SINCE)) {
      resp.sendError(HttpServletResponse.SC_NOT_MODIFIED);
      return;
    }

    // Send back the correct content type and character encoding headers
    resp.setContentType(asset.mediaType.toString());
    if (asset.mediaType.charset().isPresent()) {
      resp.setCharacterEncoding(asset.mediaType.charset().get().toString());
    }

    // Finally write the bytes of the asset out
    ServletOutputStream output = resp.getOutputStream();
    try {
      output.write(asset.bytes);
    } finally {
      output.close();
    }
  }

  private static String getFullPath(HttpServletRequest request) {
    StringBuilder sb = new StringBuilder(request.getServletPath());
    if (request.getPathInfo() != null) {
      sb.append(request.getPathInfo());
    }

    return sb.toString();
  }

  private static String hash2etag(String hash) {
    return new EntityTag(hash).toString();
  }

  private static String etag2hash(String etag) {
    String hash;

    try {
      hash = EntityTag.valueOf(etag).getValue();
    } catch (Exception e) {
      return null;
    }

    // Jetty insists on adding a -gzip suffix to ETags sometimes.  If it's there, then strip it off.
    if (hash.endsWith("-gzip")) {
      hash = hash.substring(0, hash.length() - 5);
    }

    return hash;
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy