Gitiles is a simple browser for Git repositories

It is based on JGit and uses Google Closure Templates as a templating
language. Access to underlying repositories is based on a few simple
interfaces; currently, there is only a simple disk-based
implementation, but other implementations are possible.

Features include viewing repositories by branch, shortlogs, showing
individual files and diffs with syntax highlighting, with many more
planned.

The application itself is a standard Java servlet and is configured
primarily via a git config format file. Deploying the WAR in any
servlet container should be possible.

In addition, a standalone server may be run with jetty-maven-plugin
with `mvn package && mvn jetty:run`.

Change-Id: I0ab8875b6c50f7df03b9a42b4a60923a4827bde7
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..ebbbd5f
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,3 @@
+/.classpath
+/.project
+/.settings
diff --git a/.settings/org.eclipse.core.resources.prefs b/.settings/org.eclipse.core.resources.prefs
new file mode 100644
index 0000000..5b781ec
--- /dev/null
+++ b/.settings/org.eclipse.core.resources.prefs
@@ -0,0 +1,3 @@
+eclipse.preferences.version=1
+encoding//src/main/java=UTF-8
+encoding//src/test/java=UTF-8
diff --git a/.settings/org.eclipse.jdt.core.prefs b/.settings/org.eclipse.jdt.core.prefs
new file mode 100644
index 0000000..60105c1
--- /dev/null
+++ b/.settings/org.eclipse.jdt.core.prefs
@@ -0,0 +1,5 @@
+eclipse.preferences.version=1
+org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6
+org.eclipse.jdt.core.compiler.compliance=1.6
+org.eclipse.jdt.core.compiler.problem.forbiddenReference=warning
+org.eclipse.jdt.core.compiler.source=1.6
diff --git a/.settings/org.eclipse.jdt.ui.prefs b/.settings/org.eclipse.jdt.ui.prefs
new file mode 100644
index 0000000..fa9df30
--- /dev/null
+++ b/.settings/org.eclipse.jdt.ui.prefs
@@ -0,0 +1,5 @@
+eclipse.preferences.version=1
+org.eclipse.jdt.ui.ignorelowercasenames=true
+org.eclipse.jdt.ui.importorder=com.google;com;junit;net;org;java;javax;
+org.eclipse.jdt.ui.ondemandthreshold=999
+org.eclipse.jdt.ui.staticondemandthreshold=999
diff --git a/gitiles-servlet/pom.xml b/gitiles-servlet/pom.xml
new file mode 100644
index 0000000..5537693
--- /dev/null
+++ b/gitiles-servlet/pom.xml
@@ -0,0 +1,115 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!--
+Copyright 2012 Google Inc. All Rights Reserved.
+
+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.
+-->
+<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
+  <modelVersion>4.0.0</modelVersion>
+
+  <parent>
+    <groupId>com.google.gitiles</groupId>
+    <artifactId>gitiles-parent</artifactId>
+    <version>1.0-SNAPSHOT</version>
+  </parent>
+
+  <artifactId>gitiles-servlet</artifactId>
+  <name>Gitiles - Servlet</name>
+
+  <description>
+    Gitiles servlet implementation
+  </description>
+
+  <dependencies>
+    <dependency>
+      <groupId>com.google.guava</groupId>
+      <artifactId>guava</artifactId>
+    </dependency>
+
+    <dependency>
+      <groupId>com.google.template</groupId>
+      <artifactId>soy</artifactId>
+    </dependency>
+
+    <dependency>
+      <groupId>org.eclipse.jgit</groupId>
+      <artifactId>org.eclipse.jgit</artifactId>
+    </dependency>
+
+    <dependency>
+      <groupId>org.eclipse.jgit</groupId>
+      <artifactId>org.eclipse.jgit.http.server</artifactId>
+    </dependency>
+
+    <dependency>
+      <groupId>org.eclipse.jgit</groupId>
+      <artifactId>org.eclipse.jgit.junit</artifactId>
+      <scope>test</scope>
+      <exclusions>
+        <exclusion>
+          <groupId>org.eclipse.jgit</groupId>
+          <artifactId>org.eclipse.jgit</artifactId>
+        </exclusion>
+      </exclusions>
+    </dependency>
+
+    <dependency>
+      <groupId>org.apache.tomcat</groupId>
+      <artifactId>servlet-api</artifactId>
+      <scope>provided</scope>
+    </dependency>
+
+    <dependency>
+      <groupId>junit</groupId>
+      <artifactId>junit</artifactId>
+      <scope>test</scope>
+    </dependency>
+
+    <dependency>
+      <groupId>org.slf4j</groupId>
+      <artifactId>slf4j-api</artifactId>
+    </dependency>
+
+    <dependency>
+      <groupId>joda-time</groupId>
+      <artifactId>joda-time</artifactId>
+    </dependency>
+
+    <dependency>
+      <groupId>com.google.code.gson</groupId>
+      <artifactId>gson</artifactId>
+    </dependency>
+
+    <dependency>
+      <groupId>org.apache.commons</groupId>
+      <artifactId>commons-lang3</artifactId>
+    </dependency>
+  </dependencies>
+
+  <build>
+    <plugins>
+      <plugin>
+        <groupId>org.apache.maven.plugins</groupId>
+        <artifactId>maven-source-plugin</artifactId>
+        <executions>
+          <execution>
+            <goals>
+              <goal>jar</goal>
+            </goals>
+          </execution>
+        </executions>
+      </plugin>
+    </plugins>
+  </build>
+</project>
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/AbstractHttpFilter.java b/gitiles-servlet/src/main/java/com/google/gitiles/AbstractHttpFilter.java
new file mode 100644
index 0000000..092e12d
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/AbstractHttpFilter.java
@@ -0,0 +1,49 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import java.io.IOException;
+
+import javax.servlet.Filter;
+import javax.servlet.FilterChain;
+import javax.servlet.FilterConfig;
+import javax.servlet.ServletException;
+import javax.servlet.ServletRequest;
+import javax.servlet.ServletResponse;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+abstract class AbstractHttpFilter implements Filter {
+  @Override
+  public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain)
+      throws IOException, ServletException {
+    doFilter((HttpServletRequest) req, (HttpServletResponse) res, chain);
+  }
+
+  @Override
+  @SuppressWarnings("unused") // Allow subclasses to throw ServletException.
+  public void init(FilterConfig config) throws ServletException {
+    // Default implementation does nothing.
+  }
+
+  @Override
+  public void destroy() {
+    // Default implementation does nothing.
+  }
+
+  /** @see #doFilter(ServletRequest, ServletResponse, FilterChain) */
+  public abstract void doFilter(HttpServletRequest req, HttpServletResponse res, FilterChain chain)
+      throws IOException, ServletException;
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/BaseServlet.java b/gitiles-servlet/src/main/java/com/google/gitiles/BaseServlet.java
new file mode 100644
index 0000000..70dc14e
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/BaseServlet.java
@@ -0,0 +1,119 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static javax.servlet.http.HttpServletResponse.SC_NOT_FOUND;
+
+import com.google.common.base.Charsets;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.Maps;
+import com.google.common.net.HttpHeaders;
+
+import org.joda.time.Instant;
+
+import java.io.IOException;
+import java.util.Map;
+
+import javax.servlet.http.HttpServlet;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Base servlet class for Gitiles servlets that serve Soy templates. */
+public abstract class BaseServlet extends HttpServlet {
+  private static final String DATA_ATTRIBUTE = BaseServlet.class.getName() + "/Data";
+
+  static void setNotCacheable(HttpServletResponse res) {
+    res.setHeader(HttpHeaders.CACHE_CONTROL, "no-cache, no-store, max-age=0, must-revalidate");
+    res.setHeader(HttpHeaders.PRAGMA, "no-cache");
+    res.setHeader(HttpHeaders.EXPIRES, "Fri, 01 Jan 1990 00:00:00 GMT");
+    res.setDateHeader(HttpHeaders.DATE, new Instant().getMillis());
+  }
+
+  public static BaseServlet notFoundServlet() {
+    return new BaseServlet(null) {
+      @Override
+      public void service(HttpServletRequest req, HttpServletResponse res) {
+        res.setStatus(SC_NOT_FOUND);
+      }
+    };
+  }
+
+  public static Map<String, String> menuEntry(String text, String url) {
+    if (url != null) {
+      return ImmutableMap.of("text", text, "url", url);
+    } else {
+      return ImmutableMap.of("text", text);
+    }
+  }
+
+  protected static Map<String, Object> getData(HttpServletRequest req) {
+    @SuppressWarnings("unchecked")
+    Map<String, Object> data = (Map<String, Object>) req.getAttribute(DATA_ATTRIBUTE);
+    if (data == null) {
+      data = Maps.newHashMap();
+      req.setAttribute(DATA_ATTRIBUTE, data);
+    }
+    return data;
+  }
+
+  protected final Renderer renderer;
+
+  protected BaseServlet(Renderer renderer) {
+    this.renderer = renderer;
+  }
+
+  /**
+   * Put a value into a request's Soy data map.
+   * <p>
+   * This method is intended to support a composition pattern whereby a
+   * {@link BaseServlet} is wrapped in a different {@link HttpServlet} that can
+   * update its data map.
+   *
+   * @param req in-progress request.
+   * @param key key.
+   * @param value Soy data value.
+   */
+  public void put(HttpServletRequest req, String key, Object value) {
+    getData(req).put(key, value);
+  }
+
+  protected void render(HttpServletRequest req, HttpServletResponse res, String templateName,
+      Map<String, ?> soyData) throws IOException {
+    try {
+      res.setContentType(FormatType.HTML.getMimeType());
+      res.setCharacterEncoding(Charsets.UTF_8.name());
+      setCacheHeaders(req, res);
+
+      Map<String, Object> allData = getData(req);
+      allData.putAll(soyData);
+      GitilesView view = ViewFilter.getView(req);
+      if (!allData.containsKey("repositoryName") && view.getRepositoryName() != null) {
+        allData.put("repositoryName", view.getRepositoryName());
+      }
+      if (!allData.containsKey("breadcrumbs")) {
+        allData.put("breadcrumbs", view.getBreadcrumbs());
+      }
+
+      res.setStatus(HttpServletResponse.SC_OK);
+      renderer.render(res, templateName, allData);
+    } finally {
+      req.removeAttribute(DATA_ATTRIBUTE);
+    }
+  }
+
+  protected void setCacheHeaders(HttpServletRequest req, HttpServletResponse res) {
+    setNotCacheable(res);
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/BlobSoyData.java b/gitiles-servlet/src/main/java/com/google/gitiles/BlobSoyData.java
new file mode 100644
index 0000000..6b57cbc
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/BlobSoyData.java
@@ -0,0 +1,109 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static org.eclipse.jgit.lib.Constants.OBJ_COMMIT;
+
+import com.google.common.collect.Maps;
+
+import org.eclipse.jgit.diff.RawText;
+import org.eclipse.jgit.errors.LargeObjectException;
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.lib.Constants;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.lib.ObjectLoader;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.util.RawParseUtils;
+
+import java.io.IOException;
+import java.util.Map;
+
+/** Soy data converter for git blobs. */
+public class BlobSoyData {
+  /**
+   * Maximum number of bytes to load from a supposed text file for display.
+   * Files larger than this will be displayed as binary files, even if the
+   * contents was text. For example really big XML files may be above this limit
+   * and will get displayed as binary.
+   */
+  private static final int MAX_FILE_SIZE = 10 << 20;
+
+  private final GitilesView view;
+  private final RevWalk walk;
+
+  public BlobSoyData(RevWalk walk, GitilesView view) {
+    this.view = view;
+    this.walk = walk;
+  }
+
+  public Map<String, Object> toSoyData(ObjectId blobId)
+      throws MissingObjectException, IOException {
+    return toSoyData(null, blobId);
+  }
+
+  public Map<String, Object> toSoyData(String path, ObjectId blobId)
+      throws MissingObjectException, IOException {
+    Map<String, Object> data = Maps.newHashMapWithExpectedSize(4);
+    data.put("sha", ObjectId.toString(blobId));
+
+    ObjectLoader loader = walk.getObjectReader().open(blobId, Constants.OBJ_BLOB);
+    String content;
+    try {
+      byte[] raw = loader.getCachedBytes(MAX_FILE_SIZE);
+      content = !RawText.isBinary(raw) ? RawParseUtils.decode(raw) : null;
+    } catch (LargeObjectException.OutOfMemory e) {
+      throw e;
+    } catch (LargeObjectException e) {
+      content = null;
+    }
+
+    data.put("data", content);
+    if (content != null) {
+      data.put("lang", guessPrettifyLang(path, content));
+    } else if (content == null) {
+      data.put("size", Long.toString(loader.getSize()));
+    }
+    if (path != null && view.getRevision().getPeeledType() == OBJ_COMMIT) {
+      data.put("logUrl", GitilesView.log().copyFrom(view).toUrl());
+    }
+    return data;
+  }
+
+  private static String guessPrettifyLang(String path, String content) {
+    if (content.startsWith("#!/bin/sh") || content.startsWith("#!/bin/bash")) {
+      return "sh";
+    } else if (content.startsWith("#!/usr/bin/perl")) {
+      return "perl";
+    } else if (content.startsWith("#!/usr/bin/python")) {
+      return "py";
+    } else if (path == null) {
+      return null;
+    }
+
+    int slash = path.lastIndexOf('/');
+    int dot = path.lastIndexOf('.');
+    String lang = ((0 < dot) && (slash < dot)) ? path.substring(dot + 1) : null;
+    if ("txt".equalsIgnoreCase(lang)) {
+      return null;
+    } else if ("mk".equalsIgnoreCase(lang)) {
+      return "sh";
+    } else if ("Makefile".equalsIgnoreCase(path)
+        || ((0 < slash) && "Makefile".equalsIgnoreCase(path.substring(slash + 1)))) {
+      return "sh";
+    } else {
+      return lang;
+    }
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/CommitSoyData.java b/gitiles-servlet/src/main/java/com/google/gitiles/CommitSoyData.java
new file mode 100644
index 0000000..9f8a97d
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/CommitSoyData.java
@@ -0,0 +1,287 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+import static com.google.common.base.Preconditions.checkState;
+
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.ImmutableSet;
+import com.google.common.collect.Lists;
+import com.google.common.collect.Maps;
+import com.google.template.soy.data.restricted.NullData;
+
+import org.eclipse.jgit.diff.DiffEntry;
+import org.eclipse.jgit.diff.DiffEntry.ChangeType;
+import org.eclipse.jgit.diff.DiffFormatter;
+import org.eclipse.jgit.lib.AnyObjectId;
+import org.eclipse.jgit.lib.Constants;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.lib.ObjectReader;
+import org.eclipse.jgit.lib.PersonIdent;
+import org.eclipse.jgit.lib.Ref;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.treewalk.AbstractTreeIterator;
+import org.eclipse.jgit.treewalk.CanonicalTreeParser;
+import org.eclipse.jgit.treewalk.EmptyTreeIterator;
+import org.eclipse.jgit.util.GitDateFormatter;
+import org.eclipse.jgit.util.GitDateFormatter.Format;
+import org.eclipse.jgit.util.RelativeDateFormatter;
+import org.eclipse.jgit.util.io.NullOutputStream;
+
+import java.io.IOException;
+import java.util.Collections;
+import java.util.Comparator;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+import javax.annotation.Nullable;
+import javax.servlet.http.HttpServletRequest;
+
+/** Soy data converter for git commits. */
+public class CommitSoyData {
+  /** Valid sets of keys to include in Soy data for commits. */
+  public static enum KeySet {
+    DETAIL("author", "committer", "sha", "tree", "treeUrl", "parents", "message", "logUrl"),
+    DETAIL_DIFF_TREE(DETAIL, "diffTree"),
+    SHORTLOG("abbrevSha", "url", "shortMessage", "author", "branches", "tags"),
+    DEFAULT(DETAIL);
+
+    private final Set<String> keys;
+
+    private KeySet(String... keys) {
+      this.keys = ImmutableSet.copyOf(keys);
+    }
+
+    private KeySet(KeySet other, String... keys) {
+      this.keys = ImmutableSet.<String> builder().addAll(other.keys).add(keys).build();
+    }
+  }
+
+  private final Linkifier linkifier;
+  private final HttpServletRequest req;
+  private final Repository repo;
+  private final RevWalk walk;
+  private final GitilesView view;
+  private final Map<AnyObjectId, Set<Ref>> refsById;
+  private final GitDateFormatter dateFormatter;
+
+  // TODO(dborowitz): This constructor is getting a bit ridiculous.
+  public CommitSoyData(@Nullable Linkifier linkifier, HttpServletRequest req, Repository repo,
+      RevWalk walk, GitilesView view) {
+    this(linkifier, req, repo, walk, view, null);
+  }
+
+  public CommitSoyData(@Nullable Linkifier linkifier, HttpServletRequest req, Repository repo,
+      RevWalk walk, GitilesView view, @Nullable Map<AnyObjectId, Set<Ref>> refsById) {
+    this.linkifier = linkifier;
+    this.req = req;
+    this.repo = repo;
+    this.walk = walk;
+    this.view = view;
+    this.refsById = refsById;
+    this.dateFormatter = new GitDateFormatter(Format.DEFAULT);
+  }
+
+  public Map<String, Object> toSoyData(RevCommit commit, KeySet keys) throws IOException {
+    Map<String, Object> data = Maps.newHashMapWithExpectedSize(KeySet.DEFAULT.keys.size());
+    if (keys.keys.contains("author")) {
+      data.put("author", toSoyData(commit.getAuthorIdent(), dateFormatter));
+    }
+    if (keys.keys.contains("committer")) {
+      data.put("committer", toSoyData(commit.getCommitterIdent(), dateFormatter));
+    }
+    if (keys.keys.contains("sha")) {
+      data.put("sha", ObjectId.toString(commit));
+    }
+    if (keys.keys.contains("abbrevSha")) {
+      ObjectReader reader = repo.getObjectDatabase().newReader();
+      try {
+        data.put("abbrevSha", reader.abbreviate(commit).name());
+      } finally {
+        reader.release();
+      }
+    }
+    if (keys.keys.contains("url")) {
+      data.put("url", GitilesView.revision()
+          .copyFrom(view)
+          .setRevision(commit)
+          .toUrl());
+    }
+    if (keys.keys.contains("logUrl")) {
+      Revision rev = view.getRevision();
+      GitilesView.Builder logView = GitilesView.log()
+          .copyFrom(view)
+          .setRevision(rev.getId().equals(commit) ? rev.getName() : commit.name(), commit)
+          .setOldRevision(Revision.NULL)
+          .setTreePath(null);
+      data.put("logUrl", logView.toUrl());
+    }
+    if (keys.keys.contains("tree")) {
+      data.put("tree", ObjectId.toString(commit.getTree()));
+    }
+    if (keys.keys.contains("treeUrl")) {
+      data.put("treeUrl", GitilesView.path().copyFrom(view).setTreePath("/").toUrl());
+    }
+    if (keys.keys.contains("parents")) {
+      data.put("parents", toSoyData(view, commit.getParents()));
+    }
+    if (keys.keys.contains("shortMessage")) {
+      data.put("shortMessage", commit.getShortMessage());
+    }
+    if (keys.keys.contains("branches")) {
+      data.put("branches", getRefsById(commit, Constants.R_HEADS));
+    }
+    if (keys.keys.contains("tags")) {
+      data.put("tags", getRefsById(commit, Constants.R_TAGS));
+    }
+    if (keys.keys.contains("message")) {
+      if (linkifier != null) {
+        data.put("message", linkifier.linkify(req, commit.getFullMessage()));
+      } else {
+        data.put("message", commit.getFullMessage());
+      }
+    }
+    if (keys.keys.contains("diffTree")) {
+      data.put("diffTree", computeDiffTree(commit));
+    }
+    checkState(keys.keys.size() == data.size(), "bad commit data keys: %s != %s", keys.keys,
+        data.keySet());
+    return ImmutableMap.copyOf(data);
+  }
+
+  public Map<String, Object> toSoyData(RevCommit commit) throws IOException {
+    return toSoyData(commit, KeySet.DEFAULT);
+  }
+
+  // TODO(dborowitz): Extract this.
+  static Map<String, String> toSoyData(PersonIdent ident, GitDateFormatter dateFormatter) {
+    return ImmutableMap.of(
+        "name", ident.getName(),
+        "email", ident.getEmailAddress(),
+        "time", dateFormatter.formatDate(ident),
+        // TODO(dborowitz): Switch from relative to absolute at some threshold.
+        "relativeTime", RelativeDateFormatter.format(ident.getWhen()));
+  }
+
+  private List<Map<String, String>> toSoyData(GitilesView view, RevCommit[] parents) {
+    List<Map<String, String>> result = Lists.newArrayListWithCapacity(parents.length);
+    int i = 1;
+    // TODO(dborowitz): Render something slightly different when we're actively
+    // viewing a diff against one of the parents.
+    for (RevCommit parent : parents) {
+      String name = parent.name();
+      GitilesView.Builder diff = GitilesView.diff().copyFrom(view).setTreePath("");
+      String parentName;
+      if (parents.length == 1) {
+        parentName = view.getRevision().getName() + "^";
+      } else {
+        parentName = view.getRevision().getName() + "^" + (i++);
+      }
+      result.add(ImmutableMap.of(
+          "sha", name,
+          "url", GitilesView.revision()
+              .copyFrom(view)
+              .setRevision(parentName, parent)
+              .toUrl(),
+          "diffUrl", diff.setOldRevision(parentName, parent).toUrl()));
+    }
+    return result;
+  }
+
+  private AbstractTreeIterator getTreeIterator(RevWalk walk, RevCommit commit) throws IOException {
+    CanonicalTreeParser p = new CanonicalTreeParser();
+    p.reset(walk.getObjectReader(), walk.parseTree(walk.parseCommit(commit).getTree()));
+    return p;
+  }
+
+  private Object computeDiffTree(RevCommit commit) throws IOException {
+    AbstractTreeIterator oldTree;
+    GitilesView.Builder diffUrl = GitilesView.diff().copyFrom(view)
+        .setTreePath("");
+    switch (commit.getParentCount()) {
+      case 0:
+        oldTree = new EmptyTreeIterator();
+        diffUrl.setOldRevision(Revision.NULL);
+        break;
+      case 1:
+        oldTree = getTreeIterator(walk, commit.getParent(0));
+        diffUrl.setOldRevision(view.getRevision().getName() + "^", commit.getParent(0));
+        break;
+      default:
+        // TODO(dborowitz): handle merges
+        return NullData.INSTANCE;
+    }
+    AbstractTreeIterator newTree = getTreeIterator(walk, commit);
+
+    DiffFormatter diff = new DiffFormatter(NullOutputStream.INSTANCE);
+    try {
+      diff.setRepository(repo);
+      diff.setDetectRenames(true);
+
+      List<Object> result = Lists.newArrayList();
+      for (DiffEntry e : diff.scan(oldTree, newTree)) {
+        Map<String, Object> entry = Maps.newHashMapWithExpectedSize(5);
+        entry.put("path", e.getNewPath());
+        entry.put("url", GitilesView.path()
+            .copyFrom(view)
+            .setTreePath(e.getNewPath())
+            .toUrl());
+        entry.put("diffUrl", diffUrl.setAnchor("F" + result.size()).toUrl());
+        entry.put("changeType", e.getChangeType().toString());
+        if (e.getChangeType() == ChangeType.COPY || e.getChangeType() == ChangeType.RENAME) {
+          entry.put("oldPath", e.getOldPath());
+        }
+        result.add(entry);
+      }
+      return result;
+    } finally {
+      diff.release();
+    }
+  }
+
+  private static final Comparator<Map<String, String>> NAME_COMPARATOR =
+      new Comparator<Map<String, String>>() {
+        @Override
+        public int compare(Map<String, String> o1, Map<String, String> o2) {
+          return o1.get("name").compareTo(o2.get("name"));
+        }
+      };
+
+  private List<Map<String, String>> getRefsById(ObjectId id, String prefix) {
+    checkNotNull(refsById, "must pass in ID to ref map to look up refs by ID");
+    Set<Ref> refs = refsById.get(id);
+    if (refs == null) {
+      return ImmutableList.of();
+    }
+    List<Map<String, String>> result = Lists.newArrayListWithCapacity(refs.size());
+    for (Ref ref : refs) {
+      if (ref.getName().startsWith(prefix)) {
+        result.add(ImmutableMap.of(
+          "name", ref.getName().substring(prefix.length()),
+          "url", GitilesView.revision()
+              .copyFrom(view)
+              .setRevision(Revision.unpeeled(ref.getName(), ref.getObjectId()))
+              .toUrl()));
+      }
+    }
+    Collections.sort(result, NAME_COMPARATOR);
+    return result;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/ConfigUtil.java b/gitiles-servlet/src/main/java/com/google/gitiles/ConfigUtil.java
new file mode 100644
index 0000000..4f8733b
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/ConfigUtil.java
@@ -0,0 +1,141 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import com.google.common.base.Predicates;
+import com.google.common.cache.CacheBuilder;
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.Iterables;
+
+import org.eclipse.jgit.lib.Config;
+import org.joda.time.Duration;
+
+import java.util.concurrent.TimeUnit;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+/** Utilities for working with {@link Config} objects. */
+public class ConfigUtil {
+  /**
+   * Read a duration value from the configuration.
+   * <p>
+   * Durations can be written as expressions, for example {@code "1 s"} or
+   * {@code "5 days"}. If units are not specified, milliseconds are assumed.
+   *
+   * @param config JGit config object.
+   * @param section section to read, e.g. "google"
+   * @param subsection subsection to read, e.g. "bigtable"
+   * @param name variable to read, e.g. "deadline".
+   * @param defaultValue value to use when the value is not assigned.
+   * @return a standard duration representing the time read, or defaultValue.
+   */
+  public static Duration getDuration(Config config, String section, String subsection, String name,
+      Duration defaultValue) {
+    String valStr = config.getString(section, subsection, name);
+    if (valStr == null) {
+        return defaultValue;
+    }
+
+    valStr = valStr.trim();
+    if (valStr.length() == 0) {
+        return defaultValue;
+    }
+
+    Matcher m = matcher("^([1-9][0-9]*(?:\\.[0-9]*)?)\\s*(.*)$", valStr);
+    if (!m.matches()) {
+      String key = section + (subsection != null ? "." + subsection : "") + "." + name;
+      throw new IllegalStateException("Not time unit: " + key + " = " + valStr);
+    }
+
+    String digits = m.group(1);
+    String unitName = m.group(2).trim();
+
+    TimeUnit unit;
+    if ("".equals(unitName)) {
+      unit = TimeUnit.MILLISECONDS;
+    } else if (anyOf(unitName, "ms", "millis", "millisecond", "milliseconds")) {
+      unit = TimeUnit.MILLISECONDS;
+    } else if (anyOf(unitName, "s", "sec", "second", "seconds")) {
+      unit = TimeUnit.SECONDS;
+    } else if (anyOf(unitName, "m", "min", "minute", "minutes")) {
+      unit = TimeUnit.MINUTES;
+    } else if (anyOf(unitName, "h", "hr", "hour", "hours")) {
+      unit = TimeUnit.HOURS;
+    } else if (anyOf(unitName, "d", "day", "days")) {
+      unit = TimeUnit.DAYS;
+    } else {
+      String key = section + (subsection != null ? "." + subsection : "") + "." + name;
+      throw new IllegalStateException("Not time unit: " + key + " = " + valStr);
+    }
+
+    try {
+      if (digits.indexOf('.') == -1) {
+        long val = Long.parseLong(digits);
+        return new Duration(val * TimeUnit.MILLISECONDS.convert(1, unit));
+      } else {
+        double val = Double.parseDouble(digits);
+        return new Duration((long) (val * TimeUnit.MILLISECONDS.convert(1, unit)));
+      }
+    } catch (NumberFormatException nfe) {
+      String key = section + (subsection != null ? "." + subsection : "") + "." + name;
+      throw new IllegalStateException("Not time unit: " + key + " = " + valStr, nfe);
+    }
+  }
+
+  /**
+   * Get a {@link CacheBuilder} from a config.
+   *
+   * @param config JGit config object.
+   * @param name name of the cache subsection under the "cache" section.
+   * @return a new cache builder.
+   */
+  public static CacheBuilder<Object, Object> getCacheBuilder(Config config, String name) {
+    CacheBuilder<Object, Object> b = CacheBuilder.newBuilder();
+    try {
+      if (config.getString("cache", name, "maximumWeight") != null) {
+        b.maximumWeight(config.getLong("cache", name, "maximumWeight", 20 << 20));
+      }
+      if (config.getString("cache", name, "maximumSize") != null) {
+        b.maximumSize(config.getLong("cache", name, "maximumSize", 16384));
+      }
+      Duration expireAfterWrite = getDuration(config, "cache", name, "expireAfterWrite", null);
+      if (expireAfterWrite != null) {
+        b.expireAfterWrite(expireAfterWrite.getMillis(), TimeUnit.MILLISECONDS);
+      }
+      Duration expireAfterAccess = getDuration(config, "cache", name, "expireAfterAccess", null);
+      if (expireAfterAccess != null) {
+        b.expireAfterAccess(expireAfterAccess.getMillis(), TimeUnit.MILLISECONDS);
+      }
+      // Add other methods as needed.
+    } catch (IllegalArgumentException e) {
+      throw new IllegalArgumentException("Error getting CacheBuilder for " + name, e);
+    } catch (IllegalStateException e) {
+      throw new IllegalStateException("Error getting CacheBuilder for " + name, e);
+    }
+    return b;
+  }
+
+  private static Matcher matcher(String pattern, String valStr) {
+      return Pattern.compile(pattern).matcher(valStr);
+  }
+
+  private static boolean anyOf(String a, String... cases) {
+    return Iterables.any(ImmutableList.copyOf(cases),
+        Predicates.equalTo(a.toLowerCase()));
+  }
+
+  private ConfigUtil() {
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/DebugRenderer.java b/gitiles-servlet/src/main/java/com/google/gitiles/DebugRenderer.java
new file mode 100644
index 0000000..e56518a
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/DebugRenderer.java
@@ -0,0 +1,57 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkState;
+
+import com.google.common.base.Function;
+import com.google.common.collect.ImmutableMap;
+import com.google.template.soy.SoyFileSet;
+import com.google.template.soy.tofu.SoyTofu;
+
+import java.io.File;
+import java.net.URISyntaxException;
+import java.net.URL;
+
+/** Renderer that reloads Soy templates from the filesystem on every request. */
+public class DebugRenderer extends Renderer {
+  public DebugRenderer(String staticPrefix, String customTemplatesFilename,
+      final String soyTemplatesRoot) {
+    super(
+        new Function<String, URL>() {
+          @Override
+          public URL apply(String name) {
+            return toFileURL(soyTemplatesRoot + File.separator + name);
+          }
+        },
+        ImmutableMap.<String, String> of(), staticPrefix,
+        toFileURL(customTemplatesFilename));
+  }
+
+  @Override
+  protected SoyTofu getTofu() {
+    SoyFileSet.Builder builder = new SoyFileSet.Builder()
+        .setCompileTimeGlobals(globals);
+    for (URL template : templates) {
+      try {
+        checkState(new File(template.toURI()).exists(), "Missing Soy template %s", template);
+      } catch (URISyntaxException e) {
+        throw new IllegalStateException(e);
+      }
+      builder.add(template);
+    }
+    return builder.build().compileToTofu();
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/DefaultAccess.java b/gitiles-servlet/src/main/java/com/google/gitiles/DefaultAccess.java
new file mode 100644
index 0000000..5b6d822
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/DefaultAccess.java
@@ -0,0 +1,239 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+
+import com.google.common.collect.Lists;
+import com.google.common.collect.Maps;
+import com.google.common.collect.Queues;
+
+import org.eclipse.jgit.errors.ConfigInvalidException;
+import org.eclipse.jgit.errors.RepositoryNotFoundException;
+import org.eclipse.jgit.http.server.ServletUtils;
+import org.eclipse.jgit.lib.Ref;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.lib.StoredConfig;
+import org.eclipse.jgit.transport.resolver.FileResolver;
+import org.eclipse.jgit.transport.resolver.ServiceNotEnabledException;
+import org.eclipse.jgit.util.IO;
+
+import java.io.File;
+import java.io.IOException;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.List;
+import java.util.Map;
+import java.util.Queue;
+import java.util.Set;
+
+import javax.servlet.http.HttpServletRequest;
+
+/**
+ * Default implementation of {@link GitilesAccess} with local repositories.
+ * <p>
+ * Repositories are scanned on-demand under the given path, configured by
+ * default from {@code gitiles.basePath}. There is no access control beyond what
+ * user the JVM is running under.
+ */
+public class DefaultAccess implements GitilesAccess {
+  private static final String ANONYMOUS_USER_KEY = "anonymous user";
+
+  public static class Factory implements GitilesAccess.Factory {
+    private final File basePath;
+    private final String canonicalBasePath;
+    private final String baseGitUrl;
+    private final FileResolver<HttpServletRequest> resolver;
+
+    Factory(File basePath, String baseGitUrl, FileResolver<HttpServletRequest> resolver)
+        throws IOException {
+      this.basePath = checkNotNull(basePath, "basePath");
+      this.baseGitUrl = checkNotNull(baseGitUrl, "baseGitUrl");
+      this.resolver = checkNotNull(resolver, "resolver");
+      this.canonicalBasePath = basePath.getCanonicalPath();
+    }
+
+    @Override
+    public GitilesAccess forRequest(HttpServletRequest req) {
+      String path = req.getPathInfo();
+      String repositoryPath;
+      if (path == null || path == "/") {
+        repositoryPath = null;
+      } else {
+        int slashPlus = path.indexOf("/+/");
+        if (slashPlus >= 0) {
+          repositoryPath = path.substring(0, slashPlus);
+        } else if (path.endsWith("/+")) {
+          repositoryPath = path.substring(0, path.length() - 2);
+        } else {
+          repositoryPath = path;
+        }
+      }
+      return newAccess(basePath, canonicalBasePath, baseGitUrl, resolver, req);
+    }
+
+    protected DefaultAccess newAccess(File basePath, String canonicalBasePath, String baseGitUrl,
+        FileResolver<HttpServletRequest> resolver, HttpServletRequest req) {
+      return new DefaultAccess(basePath, canonicalBasePath, baseGitUrl, resolver, req);
+    }
+  }
+
+  protected final File basePath;
+  protected final String canonicalBasePath;
+  protected final String baseGitUrl;
+  protected final FileResolver<HttpServletRequest> resolver;
+  protected final HttpServletRequest req;
+
+  protected DefaultAccess(File basePath, String canonicalBasePath, String baseGitUrl,
+      FileResolver<HttpServletRequest> resolver, HttpServletRequest req) {
+    this.basePath = checkNotNull(basePath, "basePath");
+    this.canonicalBasePath = checkNotNull(canonicalBasePath, "canonicalBasePath");
+    this.baseGitUrl = checkNotNull(baseGitUrl, "baseGitUrl");
+    this.resolver = checkNotNull(resolver, "resolver");
+    this.req = checkNotNull(req, "req");
+  }
+
+  @Override
+  public Map<String, RepositoryDescription> listRepositories(Set<String> branches)
+      throws IOException {
+    Map<String, RepositoryDescription> repos = Maps.newTreeMap();
+    for (Repository repo : scanRepositories(basePath, req)) {
+      repos.put(getRepositoryName(repo), buildDescription(repo, branches));
+      repo.close();
+    }
+    return repos;
+  }
+
+  @Override
+  public Object getUserKey() {
+    // Always return the same anonymous user key (effectively running with the
+    // same user permissions as the JVM). Subclasses may override this behavior.
+    return ANONYMOUS_USER_KEY;
+  }
+
+  @Override
+  public String getRepositoryName() {
+    return getRepositoryName(ServletUtils.getRepository(req));
+  }
+
+  @Override
+  public RepositoryDescription getRepositoryDescription() throws IOException {
+    return buildDescription(ServletUtils.getRepository(req), Collections.<String> emptySet());
+  }
+
+  private String getRepositoryName(Repository repo) {
+    String path = getRelativePath(repo);
+    if (repo.isBare() && path.endsWith(".git")) {
+      path = path.substring(0, path.length() - 4);
+    }
+    return path;
+  }
+
+  private String getRelativePath(Repository repo) {
+    String path = repo.isBare() ? repo.getDirectory().getPath() : repo.getDirectory().getParent();
+    if (repo.isBare()) {
+      path = repo.getDirectory().getPath();
+      if (path.endsWith(".git")) {
+        path = path.substring(0, path.length() - 4);
+      }
+    } else {
+      path = repo.getDirectory().getParent();
+    }
+    return getRelativePath(path);
+  }
+
+  private String getRelativePath(String path) {
+    String base = basePath.getPath();
+    if (path.startsWith(base)) {
+      return path.substring(base.length() + 1);
+    }
+    if (path.startsWith(canonicalBasePath)) {
+      return path.substring(canonicalBasePath.length() + 1);
+    }
+    throw new IllegalStateException(String.format(
+          "Repository path %s is outside base path %s", path, base));
+  }
+
+  private String loadDescriptionText(Repository repo) throws IOException {
+    String desc = null;
+    StoredConfig config = repo.getConfig();
+    IOException configError = null;
+    try {
+      config.load();
+      desc = config.getString("gitweb", null, "description");
+    } catch (ConfigInvalidException e) {
+      configError = new IOException(e);
+    }
+    if (desc == null) {
+      File descFile = new File(repo.getDirectory(), "description");
+      if (descFile.exists()) {
+        desc = new String(IO.readFully(descFile));
+      } else if (configError != null) {
+        throw configError;
+      }
+    }
+    return desc;
+  }
+
+  private RepositoryDescription buildDescription(Repository repo, Set<String> branches)
+      throws IOException {
+    RepositoryDescription desc = new RepositoryDescription();
+    desc.name = getRepositoryName(repo);
+    desc.cloneUrl = baseGitUrl + getRelativePath(repo);
+    desc.description = loadDescriptionText(repo);
+    if (!branches.isEmpty()) {
+      desc.branches = Maps.newLinkedHashMap();
+      for (String name : branches) {
+        Ref ref = repo.getRef(normalizeRefName(name));
+        if ((ref != null) && (ref.getObjectId() != null)) {
+          desc.branches.put(name, ref.getObjectId().name());
+        }
+      }
+    }
+    return desc;
+  }
+
+  private static String normalizeRefName(String name) {
+    if (name.startsWith("refs/")) {
+      return name;
+    }
+    return "refs/heads/" + name;
+  }
+
+  private Collection<Repository> scanRepositories(final File basePath, final HttpServletRequest req) throws IOException {
+    List<Repository> repos = Lists.newArrayList();
+    Queue<File> todo = Queues.newArrayDeque();
+    File[] baseFiles = basePath.listFiles();
+    if (baseFiles == null) {
+      throw new IOException("base path is not a directory: " + basePath.getPath());
+    }
+    todo.addAll(Arrays.asList(baseFiles));
+    while (!todo.isEmpty()) {
+      File file = todo.remove();
+      try {
+        repos.add(resolver.open(req, getRelativePath(file.getPath())));
+      } catch (RepositoryNotFoundException e) {
+        File[] children = file.listFiles();
+        if (children != null) {
+          todo.addAll(Arrays.asList(children));
+        }
+      } catch (ServiceNotEnabledException e) {
+        throw new IOException(e);
+      }
+    }
+    return repos;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/DefaultRenderer.java b/gitiles-servlet/src/main/java/com/google/gitiles/DefaultRenderer.java
new file mode 100644
index 0000000..f4bd1fb
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/DefaultRenderer.java
@@ -0,0 +1,59 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import com.google.common.base.Function;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.io.Resources;
+import com.google.template.soy.SoyFileSet;
+import com.google.template.soy.tofu.SoyTofu;
+
+import java.net.URL;
+import java.util.Map;
+
+/** Renderer that precompiles Soy and uses static precompiled CSS. */
+public class DefaultRenderer extends Renderer {
+  private final SoyTofu tofu;
+
+  DefaultRenderer() {
+    this("", null);
+  }
+
+  public DefaultRenderer(String staticPrefix, URL customTemplates) {
+    this(ImmutableMap.<String, String> of(), staticPrefix, customTemplates);
+  }
+
+  public DefaultRenderer(Map<String, String> globals, String staticPrefix, URL customTemplates) {
+    super(
+        new Function<String, URL>() {
+          @Override
+          public URL apply(String name) {
+            return Resources.getResource(Renderer.class, "templates/" + name);
+          }
+        },
+        globals, staticPrefix, customTemplates);
+    SoyFileSet.Builder builder = new SoyFileSet.Builder()
+        .setCompileTimeGlobals(this.globals);
+    for (URL template : templates) {
+      builder.add(template);
+    }
+    tofu = builder.build().compileToTofu();
+  }
+
+  @Override
+  protected SoyTofu getTofu() {
+    return tofu;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/DefaultUrls.java b/gitiles-servlet/src/main/java/com/google/gitiles/DefaultUrls.java
new file mode 100644
index 0000000..6495b5d
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/DefaultUrls.java
@@ -0,0 +1,60 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+
+import java.net.InetAddress;
+import java.net.UnknownHostException;
+
+import javax.servlet.http.HttpServletRequest;
+
+/**
+ * Default implementation of {@link GitilesUrls}.
+ * <p>
+ * This implementation uses statically-configured defaults, and thus assumes
+ * that the servlet is running a single virtual host.
+ */
+class DefaultUrls implements GitilesUrls {
+  private final String canonicalHostName;
+  private final String baseGitUrl;
+  private final String baseGerritUrl;
+
+  DefaultUrls(String canonicalHostName, String baseGitUrl, String baseGerritUrl)
+      throws UnknownHostException {
+    if (canonicalHostName != null) {
+      this.canonicalHostName = canonicalHostName;
+    } else {
+      this.canonicalHostName = InetAddress.getLocalHost().getCanonicalHostName();
+    }
+    this.baseGitUrl = checkNotNull(baseGitUrl, "baseGitUrl");
+    this.baseGerritUrl = checkNotNull(baseGerritUrl, "baseGerritUrl");
+  }
+
+  @Override
+  public String getHostName(HttpServletRequest req) {
+    return canonicalHostName;
+  }
+
+  @Override
+  public String getBaseGitUrl(HttpServletRequest req) {
+    return baseGitUrl;
+  }
+
+  @Override
+  public String getBaseGerritUrl(HttpServletRequest req) {
+    return baseGerritUrl;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/DiffServlet.java b/gitiles-servlet/src/main/java/com/google/gitiles/DiffServlet.java
new file mode 100644
index 0000000..2408cd7
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/DiffServlet.java
@@ -0,0 +1,162 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+import static javax.servlet.http.HttpServletResponse.SC_NOT_FOUND;
+
+import com.google.common.base.Charsets;
+
+import org.eclipse.jgit.diff.DiffFormatter;
+import org.eclipse.jgit.errors.IncorrectObjectTypeException;
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.http.server.ServletUtils;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.treewalk.AbstractTreeIterator;
+import org.eclipse.jgit.treewalk.CanonicalTreeParser;
+import org.eclipse.jgit.treewalk.EmptyTreeIterator;
+import org.eclipse.jgit.treewalk.filter.PathFilter;
+
+import java.io.IOException;
+import java.io.OutputStream;
+import java.util.Arrays;
+import java.util.Map;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Serves an HTML page with all the diffs for a commit. */
+public class DiffServlet extends BaseServlet {
+  private static final String PLACEHOLDER = "id=\"DIFF_OUTPUT_BLOCK\"";
+
+  private final Linkifier linkifier;
+
+  public DiffServlet(Renderer renderer, Linkifier linkifier) {
+    super(renderer);
+    this.linkifier = checkNotNull(linkifier, "linkifier");
+  }
+
+  @Override
+  protected void doGet(HttpServletRequest req, HttpServletResponse res) throws IOException {
+    GitilesView view = ViewFilter.getView(req);
+    Repository repo = ServletUtils.getRepository(req);
+
+    RevWalk walk = new RevWalk(repo);
+    try {
+      boolean showCommit;
+      AbstractTreeIterator oldTree;
+      AbstractTreeIterator newTree;
+      try {
+        // If we are viewing the diff between a commit and one of its parents,
+        // include the commit detail in the rendered page.
+        showCommit = isParentOf(walk, view.getOldRevision(), view.getRevision());
+        oldTree = getTreeIterator(walk, view.getOldRevision().getId());
+        newTree = getTreeIterator(walk, view.getRevision().getId());
+      } catch (MissingObjectException e) {
+        res.setStatus(SC_NOT_FOUND);
+        return;
+      } catch (IncorrectObjectTypeException e) {
+        res.setStatus(SC_NOT_FOUND);
+        return;
+      }
+
+      Map<String, Object> data = getData(req);
+      data.put("title", "Diff - " + view.getRevisionRange());
+      if (showCommit) {
+        data.put("commit", new CommitSoyData(linkifier, req, repo, walk, view)
+            .toSoyData(walk.parseCommit(view.getRevision().getId())));
+      }
+      if (!data.containsKey("repositoryName") && (view.getRepositoryName() != null)) {
+        data.put("repositoryName", view.getRepositoryName());
+      }
+      if (!data.containsKey("breadcrumbs")) {
+        data.put("breadcrumbs", view.getBreadcrumbs());
+      }
+
+      String[] html = renderAndSplit(data);
+      res.setStatus(HttpServletResponse.SC_OK);
+      res.setContentType(FormatType.HTML.getMimeType());
+      res.setCharacterEncoding(Charsets.UTF_8.name());
+      setCacheHeaders(req, res);
+
+      OutputStream out = res.getOutputStream();
+      try {
+        out.write(html[0].getBytes(Charsets.UTF_8));
+        formatHtmlDiff(out, repo, walk, oldTree, newTree, view.getTreePath());
+        out.write(html[1].getBytes(Charsets.UTF_8));
+      } finally {
+        out.close();
+      }
+    } finally {
+      walk.release();
+    }
+  }
+
+  private static boolean isParentOf(RevWalk walk, Revision oldRevision, Revision newRevision)
+      throws MissingObjectException, IncorrectObjectTypeException, IOException {
+    RevCommit newCommit = walk.parseCommit(newRevision.getId());
+    if (newCommit.getParentCount() > 0) {
+      return Arrays.asList(newCommit.getParents()).contains(oldRevision.getId());
+    } else {
+      return oldRevision == Revision.NULL;
+    }
+  }
+
+  private String[] renderAndSplit(Map<String, Object> data) {
+    String html = renderer.newRenderer("gitiles.diffDetail")
+        .setData(data)
+        .render();
+    int id = html.indexOf(PLACEHOLDER);
+    if (id < 0) {
+      throw new IllegalStateException("Template must contain " + PLACEHOLDER);
+    }
+
+    int lt = html.lastIndexOf('<', id);
+    int gt = html.indexOf('>', id + PLACEHOLDER.length());
+    return new String[] {html.substring(0, lt), html.substring(gt + 1)};
+  }
+
+  private void formatHtmlDiff(OutputStream out,
+      Repository repo, RevWalk walk,
+      AbstractTreeIterator oldTree, AbstractTreeIterator newTree,
+      String path)
+      throws IOException {
+    DiffFormatter diff = new HtmlDiffFormatter(renderer, out);
+    try {
+      if (!path.equals("")) {
+        diff.setPathFilter(PathFilter.create(path));
+      }
+      diff.setRepository(repo);
+      diff.setDetectRenames(true);
+      diff.format(oldTree, newTree);
+    } finally {
+      diff.release();
+    }
+  }
+
+  private static AbstractTreeIterator getTreeIterator(RevWalk walk, ObjectId id)
+      throws IOException {
+    if (!id.equals(ObjectId.zeroId())) {
+      CanonicalTreeParser p = new CanonicalTreeParser();
+      p.reset(walk.getObjectReader(), walk.parseTree(id));
+      return p;
+    } else {
+      return new EmptyTreeIterator();
+    }
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/FormatType.java b/gitiles-servlet/src/main/java/com/google/gitiles/FormatType.java
new file mode 100644
index 0000000..9417a4c
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/FormatType.java
@@ -0,0 +1,76 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import com.google.common.base.Strings;
+import com.google.common.net.HttpHeaders;
+
+import javax.servlet.http.HttpServletRequest;
+
+/** Type of formatting to use in the response to the client. */
+public enum FormatType {
+  HTML("text/html"),
+  TEXT("text/plain"),
+  JSON("application/json"),
+  DEFAULT("*/*");
+
+  private static final String FORMAT_TYPE_ATTRIBUTE = FormatType.class.getName();
+
+  public static FormatType getFormatType(HttpServletRequest req) {
+    FormatType result = (FormatType) req.getAttribute(FORMAT_TYPE_ATTRIBUTE);
+    if (result != null) {
+      return result;
+    }
+
+    String format = req.getParameter("format");
+    if (format != null) {
+      for (FormatType type : FormatType.values()) {
+        if (format.equalsIgnoreCase(type.name())) {
+          return set(req, type);
+        }
+      }
+      throw new IllegalArgumentException("Invalid format " + format);
+    }
+
+    String accept = req.getHeader(HttpHeaders.ACCEPT);
+    if (Strings.isNullOrEmpty(accept)) {
+      return set(req, DEFAULT);
+    }
+
+    for (String p : accept.split("[ ,;][ ,;]*")) {
+      for (FormatType type : FormatType.values()) {
+        if (p.equals(type.mimeType)) {
+          return set(req, type != HTML ? type : DEFAULT);
+        }
+      }
+    }
+    return set(req, DEFAULT);
+  }
+
+  private static FormatType set(HttpServletRequest req, FormatType format) {
+    req.setAttribute(FORMAT_TYPE_ATTRIBUTE, format);
+    return format;
+  }
+
+  private final String mimeType;
+
+  private FormatType(String mimeType) {
+    this.mimeType = mimeType;
+  }
+
+  public String getMimeType() {
+    return mimeType;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/GitilesAccess.java b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesAccess.java
new file mode 100644
index 0000000..648d709
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesAccess.java
@@ -0,0 +1,68 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import org.eclipse.jgit.transport.resolver.ServiceNotAuthorizedException;
+import org.eclipse.jgit.transport.resolver.ServiceNotEnabledException;
+
+import java.io.IOException;
+import java.util.Map;
+import java.util.Set;
+
+import javax.servlet.http.HttpServletRequest;
+
+/**
+ * Git storage interface for Gitiles.
+ * <p>
+ * Each instance is associated with a single end-user request, which implicitly
+ * includes information about the host and repository.
+ */
+public interface GitilesAccess {
+  /** Factory for per-request access. */
+  public interface Factory {
+    public GitilesAccess forRequest(HttpServletRequest req);
+  }
+
+  /**
+   * List repositories on the host.
+   *
+   * @param branches branches to list along with each repository.
+   * @return map of repository names to descriptions.
+   * @throws ServiceNotEnabledException to trigger an HTTP 403 Forbidden
+   *     (matching behavior in {@link org.eclipse.jgit.http.server.RepositoryFilter}).
+   * @throws ServiceNotAuthorizedException to trigger an HTTP 401 Unauthorized
+   *     (matching behavior in {@link org.eclipse.jgit.http.server.RepositoryFilter}).
+   * @throws IOException if an error occurred.
+   */
+  public Map<String, RepositoryDescription> listRepositories(Set<String> branches)
+      throws ServiceNotEnabledException, ServiceNotAuthorizedException, IOException;
+
+  /**
+   * @return an opaque object that uniquely identifies the end-user making the
+   *     request, and supports {@link #equals(Object)} and {@link #hashCode()}.
+   *     Never null.
+   */
+  public Object getUserKey();
+
+  /** @return the repository name associated with the request. */
+  public String getRepositoryName();
+
+  /**
+   * @return the description attached to the repository of this request.
+   * @throws IOException an error occurred reading the description string from
+   *         the repository.
+   */
+  public RepositoryDescription getRepositoryDescription() throws IOException;
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/GitilesFilter.java b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesFilter.java
new file mode 100644
index 0000000..b2cf9ec
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesFilter.java
@@ -0,0 +1,362 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+import static com.google.common.base.Preconditions.checkState;
+import static com.google.gitiles.GitilesServlet.STATIC_PREFIX;
+import static com.google.gitiles.ViewFilter.getRegexGroup;
+
+import com.google.common.annotations.VisibleForTesting;
+import com.google.common.base.Joiner;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.LinkedListMultimap;
+import com.google.common.collect.ListMultimap;
+import com.google.common.collect.Maps;
+
+import org.eclipse.jgit.errors.ConfigInvalidException;
+import org.eclipse.jgit.errors.RepositoryNotFoundException;
+import org.eclipse.jgit.http.server.RepositoryFilter;
+import org.eclipse.jgit.http.server.glue.MetaFilter;
+import org.eclipse.jgit.lib.Config;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.storage.file.FileBasedConfig;
+import org.eclipse.jgit.transport.ServiceMayNotContinueException;
+import org.eclipse.jgit.transport.resolver.FileResolver;
+import org.eclipse.jgit.transport.resolver.RepositoryResolver;
+import org.eclipse.jgit.transport.resolver.ServiceNotAuthorizedException;
+import org.eclipse.jgit.transport.resolver.ServiceNotEnabledException;
+import org.eclipse.jgit.util.FS;
+
+import java.io.File;
+import java.io.IOException;
+import java.net.UnknownHostException;
+import java.util.Iterator;
+import java.util.Map;
+import java.util.regex.Pattern;
+
+import javax.servlet.Filter;
+import javax.servlet.FilterChain;
+import javax.servlet.FilterConfig;
+import javax.servlet.ServletException;
+import javax.servlet.ServletRequest;
+import javax.servlet.ServletResponse;
+import javax.servlet.http.HttpServlet;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/**
+ * MetaFilter to serve Gitiles.
+ * <p>
+ * Do not use directly; use {@link GitilesServlet}.
+ */
+class GitilesFilter extends MetaFilter {
+  private static final String CONFIG_PATH_PARAM = "configPath";
+
+  // The following regexes have the following capture groups:
+  // 1. The whole string, which causes RegexPipeline to set REGEX_GROUPS but
+  //    not otherwise modify the original request.
+  // 2. The repository name part, before /<CMD>.
+  // 3. The command, <CMD>, with no slashes and beginning with +. Commands have
+  //    names analogous to (but not exactly the same as) git command names, such
+  //    as "+log" and "+show". The bare command "+" maps to one of the other
+  //    commands based on the revision/path, and may change over time.
+  // 4. The revision/path part, after /<CMD> (called just "path" below). This is
+  //    split into a revision and a path by RevisionParser.
+
+  private static final String CMD = "\\+[a-z0-9-]*";
+
+  @VisibleForTesting
+  static final Pattern ROOT_REGEX = Pattern.compile(""
+      + "^(      " // 1. Everything
+      + "  /*    " // Excess slashes
+      + "  (/)   " // 2. Repo name (just slash)
+      + "  ()    " // 3. Command
+      + "  ()    " // 4. Path
+      + ")$      ",
+      Pattern.COMMENTS);
+
+  @VisibleForTesting
+  static final Pattern REPO_REGEX = Pattern.compile(""
+      + "^(                     " // 1. Everything
+      + "  /*                   " // Excess slashes
+      + "  (                    " // 2. Repo name
+      + "   /                   " // Leading slash
+      + "   (?:.(?!             " // Anything, as long as it's not followed by...
+      + "        /" + CMD + "/  " // the special "/<CMD>/" separator,
+      + "        |/" + CMD + "$ " // or "/<CMD>" at the end of the string
+      + "        ))*?           "
+      + "  )                    "
+      + "  /*                   " // Trailing slashes
+      + "  ()                   " // 3. Command
+      + "  ()                   " // 4. Path
+      + ")$                     ",
+      Pattern.COMMENTS);
+
+  @VisibleForTesting
+  static final Pattern REPO_PATH_REGEX = Pattern.compile(""
+      + "^(              " // 1. Everything
+      + "  /*            " // Excess slashes
+      + "  (             " // 2. Repo name
+      + "   /            " // Leading slash
+      + "   .*?          " // Anything, non-greedy
+      + "  )             "
+      + "  /(" + CMD + ")" // 3. Command
+      + "  (             " // 4. Path
+      + "   (?:/.*)?     " // Slash path, or nothing.
+      + "  )             "
+      + ")$              ",
+      Pattern.COMMENTS);
+
+  private static class DispatchFilter extends AbstractHttpFilter {
+    private final ListMultimap<GitilesView.Type, Filter> filters;
+    private final Map<GitilesView.Type, HttpServlet> servlets;
+
+    private DispatchFilter(ListMultimap<GitilesView.Type, Filter> filters,
+        Map<GitilesView.Type, HttpServlet> servlets) {
+      this.filters = LinkedListMultimap.create(filters);
+      this.servlets = ImmutableMap.copyOf(servlets);
+      for (GitilesView.Type type : GitilesView.Type.values()) {
+        checkState(servlets.containsKey(type), "Missing handler for view %s", type);
+      }
+    }
+
+    @Override
+    public void doFilter(HttpServletRequest req, HttpServletResponse res, FilterChain chain)
+        throws IOException, ServletException {
+      GitilesView view = checkNotNull(ViewFilter.getView(req));
+      final Iterator<Filter> itr = filters.get(view.getType()).iterator();
+      final HttpServlet servlet = servlets.get(view.getType());
+      new FilterChain() {
+        @Override
+        public void doFilter(ServletRequest req, ServletResponse res)
+            throws IOException, ServletException {
+          if (itr.hasNext()) {
+            itr.next().doFilter(req, res, this);
+          } else {
+            servlet.service(req, res);
+          }
+        }
+      }.doFilter(req, res);
+    }
+  }
+
+  private final ListMultimap<GitilesView.Type, Filter> filters = LinkedListMultimap.create();
+  private final Map<GitilesView.Type, HttpServlet> servlets = Maps.newHashMap();
+
+  private Renderer renderer;
+  private GitilesUrls urls;
+  private Linkifier linkifier;
+  private GitilesAccess.Factory accessFactory;
+  private RepositoryResolver<HttpServletRequest> resolver;
+  private VisibilityCache visibilityCache;
+  private boolean initialized;
+
+  GitilesFilter() {
+  }
+
+  GitilesFilter(
+      Renderer renderer,
+      GitilesUrls urls,
+      GitilesAccess.Factory accessFactory,
+      final RepositoryResolver<HttpServletRequest> resolver,
+      VisibilityCache visibilityCache) {
+    this.renderer = checkNotNull(renderer, "renderer");
+    this.urls = checkNotNull(urls, "urls");
+    this.accessFactory = checkNotNull(accessFactory, "accessFactory");
+    this.visibilityCache = checkNotNull(visibilityCache, "visibilityCache");
+    this.linkifier = new Linkifier(urls);
+    this.resolver = wrapResolver(resolver);
+  }
+
+  @Override
+  public synchronized void init(FilterConfig config) throws ServletException {
+    super.init(config);
+    setDefaultFields(config);
+
+    for (GitilesView.Type type : GitilesView.Type.values()) {
+      if (!servlets.containsKey(type)) {
+        servlets.put(type, getDefaultHandler(type));
+      }
+    }
+
+    Filter repositoryFilter = new RepositoryFilter(resolver);
+    Filter viewFilter = new ViewFilter(accessFactory, urls, visibilityCache);
+    Filter dispatchFilter = new DispatchFilter(filters, servlets);
+    String browserCssName;
+    String prettifyCssName;
+    String prettifyJsName;
+
+    serveRegex(ROOT_REGEX)
+        .through(viewFilter)
+        .through(dispatchFilter);
+
+    serveRegex(REPO_REGEX)
+        .through(repositoryFilter)
+        .through(viewFilter)
+        .through(dispatchFilter);
+
+    serveRegex(REPO_PATH_REGEX)
+        .through(repositoryFilter)
+        .through(viewFilter)
+        .through(dispatchFilter);
+
+    initialized = true;
+  }
+
+  public synchronized BaseServlet getDefaultHandler(GitilesView.Type view) {
+    checkNotInitialized();
+    switch (view) {
+      case HOST_INDEX:
+        return new HostIndexServlet(renderer, urls, accessFactory);
+      case REPOSITORY_INDEX:
+        return new RepositoryIndexServlet(renderer, accessFactory);
+      case REVISION:
+        return new RevisionServlet(renderer, linkifier);
+      case PATH:
+        return new PathServlet(renderer);
+      case DIFF:
+        return new DiffServlet(renderer, linkifier);
+      case LOG:
+        return new LogServlet(renderer, linkifier);
+      default:
+        throw new IllegalArgumentException("Invalid view type: " + view);
+    }
+  }
+
+  synchronized void addFilter(GitilesView.Type view, Filter filter) {
+    checkNotInitialized();
+    filters.put(checkNotNull(view, "view"), checkNotNull(filter, "filter for %s", view));
+  }
+
+  synchronized void setHandler(GitilesView.Type view, HttpServlet handler) {
+    checkNotInitialized();
+    servlets.put(checkNotNull(view, "view"),
+        checkNotNull(handler, "handler for %s", view));
+  }
+
+  private synchronized void checkNotInitialized() {
+    checkState(!initialized, "Gitiles already initialized");
+  }
+
+  private static RepositoryResolver<HttpServletRequest> wrapResolver(
+      final RepositoryResolver<HttpServletRequest> resolver) {
+    checkNotNull(resolver, "resolver");
+    return new RepositoryResolver<HttpServletRequest>() {
+      @Override
+      public Repository open(HttpServletRequest req, String name)
+          throws RepositoryNotFoundException, ServiceNotAuthorizedException,
+          ServiceNotEnabledException, ServiceMayNotContinueException {
+        return resolver.open(req, ViewFilter.trimLeadingSlash(getRegexGroup(req, 1)));
+      }
+    };
+  }
+
+  private void setDefaultFields(FilterConfig config) throws ServletException {
+    if (renderer != null && urls != null && accessFactory != null && resolver != null
+        && visibilityCache != null) {
+      return;
+    }
+    String configPath = config.getInitParameter(CONFIG_PATH_PARAM);
+    if (configPath == null) {
+      throw new ServletException("Missing required parameter " + configPath);
+    }
+    FileBasedConfig jgitConfig = new FileBasedConfig(new File(configPath), FS.DETECTED);
+    try {
+      jgitConfig.load();
+    } catch (IOException e) {
+      throw new ServletException(e);
+    } catch (ConfigInvalidException e) {
+      throw new ServletException(e);
+    }
+
+    if (renderer == null) {
+      String staticPrefix = config.getServletContext().getContextPath() + STATIC_PREFIX;
+      String customTemplates = jgitConfig.getString("gitiles", null, "customTemplates");
+      // TODO(dborowitz): Automatically set to true when run with mvn jetty:run.
+      if (jgitConfig.getBoolean("gitiles", null, "reloadTemplates", false)) {
+        renderer = new DebugRenderer(staticPrefix, customTemplates,
+            Joiner.on(File.separatorChar).join(System.getProperty("user.dir"),
+                "gitiles-servlet", "src", "main", "resources",
+                "com", "google", "gitiles", "templates"));
+      } else {
+        renderer = new DefaultRenderer(staticPrefix, Renderer.toFileURL(customTemplates));
+      }
+    }
+    if (urls == null) {
+      try {
+        urls = new DefaultUrls(
+            jgitConfig.getString("gitiles", null, "canonicalHostName"),
+            getBaseGitUrl(jgitConfig),
+            getGerritUrl(jgitConfig));
+      } catch (UnknownHostException e) {
+        throw new ServletException(e);
+      }
+    }
+    linkifier = new Linkifier(urls);
+    if (accessFactory == null || resolver == null) {
+      String basePath = jgitConfig.getString("gitiles", null, "basePath");
+      if (basePath == null) {
+        throw new ServletException("gitiles.basePath not set");
+      }
+      boolean exportAll = jgitConfig.getBoolean("gitiles", null, "exportAll", false);
+
+      FileResolver<HttpServletRequest> fileResolver;
+      if (resolver == null) {
+        fileResolver = new FileResolver<HttpServletRequest>(new File(basePath), exportAll);
+        resolver = wrapResolver(fileResolver);
+      } else if (resolver instanceof FileResolver) {
+        fileResolver = (FileResolver<HttpServletRequest>) resolver;
+      } else {
+        fileResolver = null;
+      }
+      if (accessFactory == null) {
+        checkState(fileResolver != null, "need a FileResolver when GitilesAccess.Factory not set");
+        try {
+        accessFactory = new DefaultAccess.Factory(
+            new File(basePath),
+            getBaseGitUrl(jgitConfig),
+            fileResolver);
+        } catch (IOException e) {
+          throw new ServletException(e);
+        }
+      }
+    }
+    if (visibilityCache == null) {
+      if (jgitConfig.getSubsections("cache").contains("visibility")) {
+        visibilityCache =
+            new VisibilityCache(false, ConfigUtil.getCacheBuilder(jgitConfig, "visibility"));
+      } else {
+        visibilityCache = new VisibilityCache(false);
+      }
+    }
+  }
+
+  private static String getBaseGitUrl(Config config) throws ServletException {
+    String baseGitUrl = config.getString("gitiles", null, "baseGitUrl");
+    if (baseGitUrl == null) {
+      throw new ServletException("gitiles.baseGitUrl not set");
+    }
+    return baseGitUrl;
+  }
+
+  private static String getGerritUrl(Config config) throws ServletException {
+    String gerritUrl = config.getString("gitiles", null, "gerritUrl");
+    if (gerritUrl == null) {
+      throw new ServletException("gitiles.gerritUrl not set");
+    }
+    return gerritUrl;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/GitilesServlet.java b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesServlet.java
new file mode 100644
index 0000000..2cdbc02
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesServlet.java
@@ -0,0 +1,113 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import org.eclipse.jgit.http.server.glue.MetaServlet;
+import org.eclipse.jgit.transport.resolver.RepositoryResolver;
+
+import java.util.Enumeration;
+
+import javax.servlet.Filter;
+import javax.servlet.FilterConfig;
+import javax.servlet.ServletConfig;
+import javax.servlet.ServletContext;
+import javax.servlet.ServletException;
+import javax.servlet.http.HttpServlet;
+import javax.servlet.http.HttpServletRequest;
+
+/**
+ * Servlet to serve Gitiles.
+ * <p>
+ * This servlet can either be constructed manually with its dependencies, or
+ * configured to use default implementations for the Gitiles interfaces. To
+ * configure the defaults, you must provide a single init parameter
+ * "configPath", which is the path to a git config file containing additional
+ * configuration.
+ */
+public class GitilesServlet extends MetaServlet {
+  /** The prefix from which static resources are served. */
+  public static final String STATIC_PREFIX = "/+static/";
+
+  public GitilesServlet(Renderer renderer,
+      GitilesUrls urls,
+      GitilesAccess.Factory accessFactory,
+      RepositoryResolver<HttpServletRequest> resolver,
+      VisibilityCache visibilityCache) {
+    super(new GitilesFilter(renderer, urls, accessFactory, resolver, visibilityCache));
+  }
+
+  public GitilesServlet() {
+    super(new GitilesFilter());
+  }
+
+  @Override
+  protected GitilesFilter getDelegateFilter() {
+    return (GitilesFilter) super.getDelegateFilter();
+  }
+
+  @Override
+  public void init(final ServletConfig config) throws ServletException {
+    getDelegateFilter().init(new FilterConfig() {
+      @Override
+      public String getFilterName() {
+        return getDelegateFilter().getClass().getName();
+      }
+
+      @Override
+      public String getInitParameter(String name) {
+        return config.getInitParameter(name);
+      }
+
+      @SuppressWarnings("rawtypes")
+      @Override
+      public Enumeration getInitParameterNames() {
+        return config.getInitParameterNames();
+      }
+
+      @Override
+      public ServletContext getServletContext() {
+        return config.getServletContext();
+      }
+    });
+  }
+
+  /**
+   * Add a custom filter for a view.
+   * <p>
+   * Must be called before initializing the servlet.
+   *
+   * @param view view type.
+   * @param filter filter.
+   */
+  public void addFilter(GitilesView.Type view, Filter filter) {
+    getDelegateFilter().addFilter(view, filter);
+  }
+
+  /**
+   * Set a custom handler for a view.
+   * <p>
+   * Must be called before initializing the servlet.
+   *
+   * @param view view type.
+   * @param handler handler.
+   */
+  public void setHandler(GitilesView.Type view, HttpServlet handler) {
+    getDelegateFilter().setHandler(view, handler);
+  }
+
+  public BaseServlet getDefaultHandler(GitilesView.Type view) {
+    return getDelegateFilter().getDefaultHandler(view);
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/GitilesUrls.java b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesUrls.java
new file mode 100644
index 0000000..fb7ff3b
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesUrls.java
@@ -0,0 +1,79 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import com.google.common.base.Charsets;
+import com.google.common.base.Function;
+
+import java.io.UnsupportedEncodingException;
+import java.net.URLEncoder;
+
+import javax.servlet.http.HttpServletRequest;
+
+/** Interface for URLs displayed on source browsing pages. */
+public interface GitilesUrls {
+  /**
+   * Escapes repository or path names to be safely embedded into a URL.
+   * <p>
+   * This escape implementation escapes a repository or path name such as
+   * "foo/bar</child" to appear as "foo/bar%3C/child". Spaces are escaped as
+   * "%20". Its purpose is to escape a repository name to be safe for inclusion
+   * in the path component of the URL, where "/" is a valid character that
+   * should not be encoded, while almost any other non-alpha, non-numeric
+   * character will be encoded using URL style encoding.
+   */
+  public static final Function<String, String> NAME_ESCAPER = new Function<String, String>() {
+    @Override
+    public String apply(String s) {
+      try {
+        return URLEncoder.encode(s, Charsets.UTF_8.name())
+            .replace("%2F", "/")
+            .replace("%2f", "/")
+            .replace("+", "%20")
+            .replace("%2B", "+")
+            .replace("%2b", "+");
+      } catch (UnsupportedEncodingException e) {
+        throw new IllegalStateException(e);
+      }
+    }
+  };
+
+  /**
+   * Return the name of the host from the request.
+   *
+   * Used in various user-visible text, like "MyHost Git Repositories".
+   *
+   * @param req request.
+   * @return host name; may be null.
+   */
+  public String getHostName(HttpServletRequest req);
+
+  /**
+   * Return the base URL for git repositories on this host.
+   *
+   * @param req request.
+   * @return base URL for git repositories.
+   */
+  public String getBaseGitUrl(HttpServletRequest req);
+
+  /**
+   * Return the base URL for Gerrit projects on this host.
+   *
+   * @param req request.
+   * @return base URL for Gerrit Code Review, or null if Gerrit is not
+   *     configured.
+   */
+  public String getBaseGerritUrl(HttpServletRequest req);
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/GitilesView.java b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesView.java
new file mode 100644
index 0000000..ad685c4
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/GitilesView.java
@@ -0,0 +1,563 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+import static com.google.common.base.Preconditions.checkState;
+import static com.google.gitiles.GitilesUrls.NAME_ESCAPER;
+
+import com.google.common.base.Charsets;
+import com.google.common.base.Objects;
+import com.google.common.base.Strings;
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.LinkedListMultimap;
+import com.google.common.collect.ListMultimap;
+import com.google.common.collect.Multimaps;
+
+import org.eclipse.jgit.revwalk.RevObject;
+
+import java.io.UnsupportedEncodingException;
+import java.net.URLEncoder;
+import java.util.List;
+import java.util.Map;
+
+import javax.servlet.http.HttpServletRequest;
+
+/**
+ * Information about a view in Gitiles.
+ * <p>
+ * Views are uniquely identified by a type, and dispatched to servlet types by
+ * {@link GitilesServlet}. This class contains the list of all types, as
+ * well as some methods containing basic information parsed from the URL.
+ * Construction happens in {@link ViewFilter}.
+ */
+public class GitilesView {
+  /** All the possible view types supported in the application. */
+  public static enum Type {
+    HOST_INDEX,
+    REPOSITORY_INDEX,
+    REVISION,
+    PATH,
+    DIFF,
+    LOG;
+  }
+
+  /** Builder for views. */
+  public static class Builder {
+    private final Type type;
+    private final ListMultimap<String, String> params = LinkedListMultimap.create();
+
+    private String hostName;
+    private String servletPath;
+    private String repositoryName;
+    private Revision revision = Revision.NULL;
+    private Revision oldRevision = Revision.NULL;
+    private String path;
+    private String anchor;
+
+    private Builder(Type type) {
+      this.type = type;
+    }
+
+    public Builder copyFrom(GitilesView other) {
+      hostName = other.hostName;
+      servletPath = other.servletPath;
+      switch (type) {
+        case LOG:
+        case DIFF:
+          oldRevision = other.oldRevision;
+          // Fallthrough.
+        case PATH:
+          path = other.path;
+          // Fallthrough.
+        case REVISION:
+          revision = other.revision;
+          // Fallthrough.
+        case REPOSITORY_INDEX:
+          repositoryName = other.repositoryName;
+      }
+      // Don't copy params.
+      return this;
+    }
+
+    public Builder copyFrom(HttpServletRequest req) {
+      return copyFrom(ViewFilter.getView(req));
+    }
+
+    public Builder setHostName(String hostName) {
+      this.hostName = checkNotNull(hostName);
+      return this;
+    }
+
+    public String getHostName() {
+      return hostName;
+    }
+
+    public Builder setServletPath(String servletPath) {
+      this.servletPath = checkNotNull(servletPath);
+      return this;
+    }
+
+    public String getServletPath() {
+      return servletPath;
+    }
+
+    public Builder setRepositoryName(String repositoryName) {
+      switch (type) {
+        case HOST_INDEX:
+          throw new IllegalStateException(String.format(
+              "cannot set repository name on %s view", type));
+        default:
+          this.repositoryName = checkNotNull(repositoryName);
+          return this;
+      }
+    }
+
+    public String getRepositoryName() {
+      return repositoryName;
+    }
+
+    public Builder setRevision(Revision revision) {
+      switch (type) {
+        case HOST_INDEX:
+        case REPOSITORY_INDEX:
+          throw new IllegalStateException(String.format("cannot set revision on %s view", type));
+        default:
+          this.revision = checkNotNull(revision);
+          return this;
+      }
+    }
+
+    public Builder setRevision(String name) {
+      return setRevision(Revision.named(name));
+    }
+
+    public Builder setRevision(RevObject obj) {
+      return setRevision(Revision.peeled(obj.name(), obj));
+    }
+
+    public Builder setRevision(String name, RevObject obj) {
+      return setRevision(Revision.peeled(name, obj));
+    }
+
+    public Revision getRevision() {
+      return revision;
+    }
+
+    public Builder setOldRevision(Revision revision) {
+      switch (type) {
+        case DIFF:
+        case LOG:
+          this.oldRevision = checkNotNull(revision);
+          return this;
+        default:
+          throw new IllegalStateException(
+              String.format("cannot set old revision on %s view", type));
+      }
+    }
+
+    public Builder setOldRevision(RevObject obj) {
+      return setOldRevision(Revision.peeled(obj.name(), obj));
+    }
+
+    public Builder setOldRevision(String name, RevObject obj) {
+      return setOldRevision(Revision.peeled(name, obj));
+    }
+
+    public Revision getOldRevision() {
+      return revision;
+    }
+
+    public Builder setTreePath(String path) {
+      switch (type) {
+        case PATH:
+        case DIFF:
+          this.path = maybeTrimLeadingAndTrailingSlash(checkNotNull(path));
+          return this;
+        case LOG:
+          this.path = path != null ? maybeTrimLeadingAndTrailingSlash(path) : null;
+          return this;
+        default:
+          throw new IllegalStateException(String.format("cannot set path on %s view", type));
+      }
+    }
+
+    public String getTreePath() {
+      return path;
+    }
+
+    public Builder putParam(String key, String value) {
+      params.put(key, value);
+      return this;
+    }
+
+    public Builder replaceParam(String key, String value) {
+      params.replaceValues(key, ImmutableList.of(value));
+      return this;
+    }
+
+    public Builder putAllParams(Map<String, String[]> params) {
+      for (Map.Entry<String, String[]> e : params.entrySet()) {
+        for (String v : e.getValue()) {
+          this.params.put(e.getKey(), v);
+        }
+      }
+      return this;
+    }
+
+    public ListMultimap<String, String> getParams() {
+      return params;
+    }
+
+    public Builder setAnchor(String anchor) {
+      this.anchor = anchor;
+      return this;
+    }
+
+    public String getAnchor() {
+      return anchor;
+    }
+
+    public GitilesView build() {
+      switch (type) {
+        case HOST_INDEX:
+          checkHostIndex();
+          break;
+        case REPOSITORY_INDEX:
+          checkRepositoryIndex();
+          break;
+        case REVISION:
+          checkRevision();
+          break;
+        case PATH:
+          checkPath();
+          break;
+        case DIFF:
+          checkDiff();
+          break;
+        case LOG:
+          checkLog();
+          break;
+      }
+      return new GitilesView(type, hostName, servletPath, repositoryName, revision,
+          oldRevision, path, params, anchor);
+    }
+
+    public String toUrl() {
+      return build().toUrl();
+    }
+
+    private void checkHostIndex() {
+      checkState(hostName != null, "missing hostName on %s view", type);
+      checkState(servletPath != null, "missing hostName on %s view", type);
+    }
+
+    private void checkRepositoryIndex() {
+      checkState(repositoryName != null, "missing repository name on %s view", type);
+      checkHostIndex();
+    }
+
+    private void checkRevision() {
+      checkState(revision != Revision.NULL, "missing revision on %s view", type);
+      checkRepositoryIndex();
+    }
+
+    private void checkDiff() {
+      checkPath();
+    }
+
+    private void checkLog() {
+      checkRevision();
+    }
+
+    private void checkPath() {
+      checkState(path != null, "missing path on %s view", type);
+      checkRevision();
+    }
+  }
+
+  public static Builder hostIndex() {
+    return new Builder(Type.HOST_INDEX);
+  }
+
+  public static Builder repositoryIndex() {
+    return new Builder(Type.REPOSITORY_INDEX);
+  }
+
+  public static Builder revision() {
+    return new Builder(Type.REVISION);
+  }
+
+  public static Builder path() {
+    return new Builder(Type.PATH);
+  }
+
+  public static Builder diff() {
+    return new Builder(Type.DIFF);
+  }
+
+  public static Builder log() {
+    return new Builder(Type.LOG);
+  }
+
+  private static String maybeTrimLeadingAndTrailingSlash(String str) {
+    if (str.startsWith("/")) {
+      str = str.substring(1);
+    }
+    return !str.isEmpty() && str.endsWith("/") ? str.substring(0, str.length() - 1) : str;
+  }
+
+  private final Type type;
+  private final String hostName;
+  private final String servletPath;
+  private final String repositoryName;
+  private final Revision revision;
+  private final Revision oldRevision;
+  private final String path;
+  private final ListMultimap<String, String> params;
+  private final String anchor;
+
+  private GitilesView(Type type,
+      String hostName,
+      String servletPath,
+      String repositoryName,
+      Revision revision,
+      Revision oldRevision,
+      String path,
+      ListMultimap<String, String> params,
+      String anchor) {
+    this.type = type;
+    this.hostName = hostName;
+    this.servletPath = servletPath;
+    this.repositoryName = repositoryName;
+    this.revision = Objects.firstNonNull(revision, Revision.NULL);
+    this.oldRevision = Objects.firstNonNull(oldRevision, Revision.NULL);
+    this.path = path;
+    this.params = Multimaps.unmodifiableListMultimap(params);
+    this.anchor = anchor;
+  }
+
+  public String getHostName() {
+    return hostName;
+  }
+
+  public String getServletPath() {
+    return servletPath;
+  }
+
+  public String getRepositoryName() {
+    return repositoryName;
+  }
+
+  public Revision getRevision() {
+    return revision;
+  }
+
+  public Revision getOldRevision() {
+    return oldRevision;
+  }
+
+  public String getRevisionRange() {
+    if (oldRevision == Revision.NULL) {
+      switch (type) {
+        case LOG:
+        case DIFF:
+          // For types that require two revisions, NULL indicates the empty
+          // tree/commit.
+          return revision.getName() + "^!";
+        default:
+          // For everything else NULL indicates it is not a range, just a single
+          // revision.
+          return null;
+      }
+    } else if (type == Type.DIFF && isFirstParent(revision, oldRevision)) {
+      return revision.getName() + "^!";
+    } else {
+      return oldRevision.getName() + ".." + revision.getName();
+    }
+  }
+
+  public String getTreePath() {
+    return path;
+  }
+
+  public ListMultimap<String, String> getParameters() {
+    return params;
+  }
+
+  public String getAnchor() {
+    return anchor;
+  }
+
+  public Type getType() {
+    return type;
+  }
+
+  /** @return an escaped, relative URL representing this view. */
+  public String toUrl() {
+    StringBuilder url = new StringBuilder(servletPath).append('/');
+    ListMultimap<String, String> params = this.params;
+    switch (type) {
+      case HOST_INDEX:
+        params = LinkedListMultimap.create();
+        if (!this.params.containsKey("format")) {
+          params.put("format", FormatType.HTML.toString());
+        }
+        params.putAll(this.params);
+        break;
+      case REPOSITORY_INDEX:
+        url.append(repositoryName).append('/');
+        break;
+      case REVISION:
+        url.append(repositoryName).append("/+");
+        if (!getRevision().nameIsId()) {
+          url.append("show"); // Default for /+/master is +log.
+        }
+        url.append('/').append(revision.getName());
+        break;
+      case PATH:
+        url.append(repositoryName).append("/+/").append(revision.getName()).append('/')
+            .append(path);
+        break;
+      case DIFF:
+        url.append(repositoryName).append("/+/");
+        if (isFirstParent(revision, oldRevision)) {
+          url.append(revision.getName()).append("^!");
+        } else {
+          url.append(oldRevision.getName()).append("..").append(revision.getName());
+        }
+        url.append('/').append(path);
+        break;
+      case LOG:
+        url.append(repositoryName).append("/+");
+        if (getRevision().nameIsId() || oldRevision != Revision.NULL || path != null) {
+         // Default for /+/c0ffee/(...) is +show.
+         // Default for /+/c0ffee..deadbeef(/...) is +diff.
+          url.append("log");
+        }
+        url.append('/');
+        if (oldRevision != Revision.NULL) {
+          url.append(oldRevision.getName()).append("..");
+        }
+        url.append(revision.getName());
+        if (path != null) {
+          url.append('/').append(path);
+        }
+        break;
+      default:
+        throw new IllegalStateException("Unknown view type: " + type);
+    }
+    String baseUrl = NAME_ESCAPER.apply(url.toString());
+    url = new StringBuilder();
+    if (!params.isEmpty()) {
+      url.append('?').append(paramsToString(params));
+    }
+    if (!Strings.isNullOrEmpty(anchor)) {
+      url.append('#').append(NAME_ESCAPER.apply(anchor));
+    }
+    return baseUrl + url.toString();
+  }
+
+  public List<Map<String, String>> getBreadcrumbs() {
+    String path = this.path;
+    ImmutableList.Builder<Map<String, String>> breadcrumbs = ImmutableList.builder();
+    breadcrumbs.add(breadcrumb(hostName, hostIndex().copyFrom(this)));
+    if (repositoryName != null) {
+      breadcrumbs.add(breadcrumb(repositoryName, repositoryIndex().copyFrom(this)));
+    }
+    if (type == Type.DIFF) {
+      // TODO(dborowitz): Tweak the breadcrumbs template to allow us to render
+      // separate links in "old..new".
+      breadcrumbs.add(breadcrumb(getRevisionRange(), diff().copyFrom(this).setTreePath("")));
+    } else if (type == Type.LOG) {
+      // TODO(dborowitz): Add something in the navigation area (probably not
+      // a breadcrumb) to allow switching between /+log/ and /+/.
+      if (oldRevision == Revision.NULL) {
+        breadcrumbs.add(breadcrumb(revision.getName(), log().copyFrom(this).setTreePath(null)));
+      } else {
+        breadcrumbs.add(breadcrumb(getRevisionRange(), log().copyFrom(this).setTreePath(null)));
+      }
+      path = Strings.emptyToNull(path);
+    } else if (revision != Revision.NULL) {
+      breadcrumbs.add(breadcrumb(revision.getName(), revision().copyFrom(this)));
+    }
+    if (path != null) {
+      if (type != Type.LOG) { // The "." breadcrumb would be no different for LOG.
+        breadcrumbs.add(breadcrumb(".", copyWithPath().setTreePath("")));
+      }
+      StringBuilder cur = new StringBuilder();
+      boolean first = true;
+      for (String part : RevisionParser.PATH_SPLITTER.omitEmptyStrings().split(path)) {
+        if (!first) {
+          cur.append('/');
+        } else {
+          first = false;
+        }
+        cur.append(part);
+        breadcrumbs.add(breadcrumb(part, copyWithPath().setTreePath(cur.toString())));
+      }
+    }
+    return breadcrumbs.build();
+  }
+
+  private static Map<String, String> breadcrumb(String text, Builder url) {
+    return ImmutableMap.of("text", text, "url", url.toUrl());
+  }
+
+  private Builder copyWithPath() {
+    Builder copy;
+    switch (type) {
+      case DIFF:
+        copy = diff();
+        break;
+      case LOG:
+        copy = log();
+        break;
+      default:
+        copy = path();
+        break;
+    }
+    return copy.copyFrom(this);
+  }
+
+  private static boolean isFirstParent(Revision rev1, Revision rev2) {
+    return rev2 == Revision.NULL
+        || rev2.getName().equals(rev1.getName() + "^")
+        || rev2.getName().equals(rev1.getName() + "~1");
+  }
+
+  private static String paramsToString(ListMultimap<String, String> params) {
+    try {
+    StringBuilder sb = new StringBuilder();
+    boolean first = true;
+    for (Map.Entry<String, String> e : params.entries()) {
+      if (!first) {
+        sb.append('&');
+      } else {
+        first = false;
+      }
+      sb.append(URLEncoder.encode(e.getKey(), Charsets.UTF_8.name()));
+      if (!"".equals(e.getValue())) {
+        sb.append('=')
+            .append(URLEncoder.encode(e.getValue(), Charsets.UTF_8.name()));
+      }
+    }
+    return sb.toString();
+    } catch (UnsupportedEncodingException e) {
+      throw new IllegalStateException(e);
+    }
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/HostIndexServlet.java b/gitiles-servlet/src/main/java/com/google/gitiles/HostIndexServlet.java
new file mode 100644
index 0000000..bf9ffd1
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/HostIndexServlet.java
@@ -0,0 +1,199 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+import static com.google.gitiles.FormatType.JSON;
+import static com.google.gitiles.FormatType.TEXT;
+import static javax.servlet.http.HttpServletResponse.SC_BAD_REQUEST;
+import static javax.servlet.http.HttpServletResponse.SC_FORBIDDEN;
+import static javax.servlet.http.HttpServletResponse.SC_NOT_FOUND;
+import static javax.servlet.http.HttpServletResponse.SC_SERVICE_UNAVAILABLE;
+import static javax.servlet.http.HttpServletResponse.SC_UNAUTHORIZED;
+
+import com.google.common.base.Strings;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.Sets;
+import com.google.common.net.HttpHeaders;
+import com.google.gson.FieldNamingPolicy;
+import com.google.gson.GsonBuilder;
+import com.google.gson.reflect.TypeToken;
+import com.google.template.soy.data.SoyListData;
+import com.google.template.soy.data.SoyMapData;
+
+import org.eclipse.jgit.errors.RepositoryNotFoundException;
+import org.eclipse.jgit.transport.ServiceMayNotContinueException;
+import org.eclipse.jgit.transport.resolver.ServiceNotAuthorizedException;
+import org.eclipse.jgit.transport.resolver.ServiceNotEnabledException;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.io.IOException;
+import java.io.PrintWriter;
+import java.util.Arrays;
+import java.util.Map;
+import java.util.Set;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Serves the top level index page for a Gitiles host. */
+public class HostIndexServlet extends BaseServlet {
+  private static final Logger log = LoggerFactory.getLogger(HostIndexServlet.class);
+
+  protected final GitilesUrls urls;
+  private final GitilesAccess.Factory accessFactory;
+
+  public HostIndexServlet(Renderer renderer, GitilesUrls urls,
+      GitilesAccess.Factory accessFactory) {
+    super(renderer);
+    this.urls = checkNotNull(urls, "urls");
+    this.accessFactory = checkNotNull(accessFactory, "accessFactory");
+  }
+
+  @Override
+  protected void doGet(HttpServletRequest req, HttpServletResponse res) throws IOException {
+    FormatType format;
+    try {
+      format = FormatType.getFormatType(req);
+    } catch (IllegalArgumentException err) {
+      res.sendError(SC_BAD_REQUEST);
+      return;
+    }
+
+    Set<String> branches = parseShowBranch(req);
+    Map<String, RepositoryDescription> descs;
+    try {
+      descs = accessFactory.forRequest(req).listRepositories(branches);
+    } catch (RepositoryNotFoundException e) {
+      res.sendError(SC_NOT_FOUND);
+      return;
+    } catch (ServiceNotEnabledException e) {
+      res.sendError(SC_FORBIDDEN);
+      return;
+    } catch (ServiceNotAuthorizedException e) {
+      res.sendError(SC_UNAUTHORIZED);
+      return;
+    } catch (ServiceMayNotContinueException e) {
+      // TODO(dborowitz): Show the error message to the user.
+      res.sendError(SC_FORBIDDEN);
+      return;
+    } catch (IOException err) {
+      String name = urls.getHostName(req);
+      log.warn("Cannot scan repositories" + (name != null ? "for " + name : ""), err);
+      res.sendError(SC_SERVICE_UNAVAILABLE);
+      return;
+    }
+
+    switch (format) {
+      case HTML:
+      case DEFAULT:
+      default:
+        displayHtml(req, res, descs);
+        break;
+
+      case TEXT:
+        displayText(req, res, branches, descs);
+        break;
+
+      case JSON:
+        displayJson(req, res, descs);
+        break;
+    }
+  }
+
+  private SoyMapData toSoyMapData(RepositoryDescription desc, GitilesView view) {
+    return new SoyMapData(
+        "name", desc.name,
+        "description", Strings.nullToEmpty(desc.description),
+        "url", GitilesView.repositoryIndex()
+            .copyFrom(view)
+            .setRepositoryName(desc.name)
+            .toUrl());
+  }
+
+  private void displayHtml(HttpServletRequest req, HttpServletResponse res,
+      Map<String, RepositoryDescription> descs) throws IOException {
+    SoyListData repos = new SoyListData();
+    for (RepositoryDescription desc : descs.values()) {
+      repos.add(toSoyMapData(desc, ViewFilter.getView(req)));
+    }
+
+    render(req, res, "gitiles.hostIndex", ImmutableMap.of(
+        "hostName", urls.getHostName(req),
+        "baseUrl", urls.getBaseGitUrl(req),
+        "repositories", repos));
+  }
+
+  private void displayText(HttpServletRequest req, HttpServletResponse res,
+      Set<String> branches, Map<String, RepositoryDescription> descs) throws IOException {
+    res.setContentType(TEXT.getMimeType());
+    res.setCharacterEncoding("UTF-8");
+    res.setHeader(HttpHeaders.CONTENT_DISPOSITION, "attachment");
+    setNotCacheable(res);
+
+    PrintWriter writer = res.getWriter();
+    for (RepositoryDescription repo : descs.values()) {
+      for (String name : branches) {
+        String ref = repo.branches.get(name);
+        if (ref == null) {
+          // Print stub (forty '-' symbols)
+          ref = "----------------------------------------";
+        }
+        writer.print(ref);
+        writer.print(' ');
+      }
+      writer.print(GitilesUrls.NAME_ESCAPER.apply(repo.name));
+      writer.print('\n');
+    }
+    writer.flush();
+    writer.close();
+  }
+
+  private void displayJson(HttpServletRequest req, HttpServletResponse res,
+      Map<String, RepositoryDescription> descs) throws IOException {
+    res.setContentType(JSON.getMimeType());
+    res.setCharacterEncoding("UTF-8");
+    res.setHeader(HttpHeaders.CONTENT_DISPOSITION, "attachment");
+    setNotCacheable(res);
+
+    PrintWriter writer = res.getWriter();
+    new GsonBuilder()
+        .setFieldNamingPolicy(FieldNamingPolicy.LOWER_CASE_WITH_UNDERSCORES)
+        .setPrettyPrinting()
+        .generateNonExecutableJson()
+        .create()
+        .toJson(descs,
+          new TypeToken<Map<String, RepositoryDescription>>() {}.getType(),
+          writer);
+    writer.print('\n');
+    writer.close();
+  }
+
+  private static Set<String> parseShowBranch(HttpServletRequest req) {
+    // Roughly match Gerrit Code Review's /projects/ API by supporting
+    // both show-branch and b as query parameters.
+    Set<String> branches = Sets.newLinkedHashSet();
+    String[] values = req.getParameterValues("show-branch");
+    if (values != null) {
+      branches.addAll(Arrays.asList(values));
+    }
+    values = req.getParameterValues("b");
+    if (values != null) {
+      branches.addAll(Arrays.asList(values));
+    }
+    return branches;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/HtmlDiffFormatter.java b/gitiles-servlet/src/main/java/com/google/gitiles/HtmlDiffFormatter.java
new file mode 100644
index 0000000..63b1eb8
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/HtmlDiffFormatter.java
@@ -0,0 +1,128 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+
+import com.google.common.base.Charsets;
+import com.google.common.collect.ImmutableMap;
+
+import org.apache.commons.lang3.StringEscapeUtils;
+import org.eclipse.jgit.diff.DiffEntry;
+import org.eclipse.jgit.diff.DiffFormatter;
+import org.eclipse.jgit.diff.RawText;
+import org.eclipse.jgit.patch.FileHeader;
+import org.eclipse.jgit.patch.FileHeader.PatchType;
+import org.eclipse.jgit.util.RawParseUtils;
+
+import java.io.IOException;
+import java.io.OutputStream;
+import java.util.List;
+
+/** Formats a unified format patch as UTF-8 encoded HTML. */
+final class HtmlDiffFormatter extends DiffFormatter {
+  private static final byte[] DIFF_BEGIN = "<pre class=\"diff-unified\">".getBytes(Charsets.UTF_8);
+  private static final byte[] DIFF_END = "</pre>".getBytes(Charsets.UTF_8);
+
+  private static final byte[] HUNK_BEGIN = "<span class=\"h\">".getBytes(Charsets.UTF_8);
+  private static final byte[] HUNK_END = "</span>".getBytes(Charsets.UTF_8);
+
+  private static final byte[] LINE_INSERT_BEGIN = "<span class=\"i\">".getBytes(Charsets.UTF_8);
+  private static final byte[] LINE_DELETE_BEGIN = "<span class=\"d\">".getBytes(Charsets.UTF_8);
+  private static final byte[] LINE_CHANGE_BEGIN = "<span class=\"c\">".getBytes(Charsets.UTF_8);
+  private static final byte[] LINE_END = "</span>\n".getBytes(Charsets.UTF_8);
+
+  private final Renderer renderer;
+  private int fileIndex;
+
+  HtmlDiffFormatter(Renderer renderer, OutputStream out) {
+    super(out);
+    this.renderer = checkNotNull(renderer, "renderer");
+  }
+
+  @Override
+  public void format(List<? extends DiffEntry> entries) throws IOException {
+    for (fileIndex = 0; fileIndex < entries.size(); fileIndex++) {
+      format(entries.get(fileIndex));
+    }
+  }
+
+  @Override
+  public void format(FileHeader hdr, RawText a, RawText b)
+      throws IOException {
+    int start = hdr.getStartOffset();
+    int end = hdr.getEndOffset();
+    if (!hdr.getHunks().isEmpty()) {
+      end = hdr.getHunks().get(0).getStartOffset();
+    }
+    renderHeader(RawParseUtils.decode(hdr.getBuffer(), start, end));
+
+    if (hdr.getPatchType() == PatchType.UNIFIED) {
+      getOutputStream().write(DIFF_BEGIN);
+      format(hdr.toEditList(), a, b);
+      getOutputStream().write(DIFF_END);
+    }
+  }
+
+  private void renderHeader(String header)
+      throws IOException {
+    int lf = header.indexOf('\n');
+    String first;
+    String rest;
+    if (0 <= lf) {
+      first = header.substring(0, lf);
+      rest = header.substring(lf + 1);
+    } else {
+      first = header;
+      rest = "";
+    }
+    getOutputStream().write(renderer.newRenderer("gitiles.diffHeader")
+        .setData(ImmutableMap.of("first", first, "rest", rest, "fileIndex", fileIndex))
+        .render()
+        .getBytes(Charsets.UTF_8));
+  }
+
+  @Override
+  protected void writeHunkHeader(int aStartLine, int aEndLine,
+      int bStartLine, int bEndLine) throws IOException {
+    getOutputStream().write(HUNK_BEGIN);
+    // TODO(sop): If hunk header starts including method names, escape it.
+    super.writeHunkHeader(aStartLine, aEndLine, bStartLine, bEndLine);
+    getOutputStream().write(HUNK_END);
+  }
+
+  @Override
+  protected void writeLine(char prefix, RawText text, int cur)
+      throws IOException {
+    // Manually render each line, rather than invoke a Soy template. This method
+    // can be called thousands of times in a single request. Avoid unnecessary
+    // overheads by formatting as-is.
+    OutputStream out = getOutputStream();
+    switch (prefix) {
+      case '+':
+        out.write(LINE_INSERT_BEGIN);
+        break;
+      case '-':
+        out.write(LINE_DELETE_BEGIN);
+        break;
+      case ' ':
+      default:
+        out.write(LINE_CHANGE_BEGIN);
+        break;
+    }
+    out.write(StringEscapeUtils.escapeHtml4(text.getString(cur)).getBytes(Charsets.UTF_8));
+    out.write(LINE_END);
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/Linkifier.java b/gitiles-servlet/src/main/java/com/google/gitiles/Linkifier.java
new file mode 100644
index 0000000..735e6f5
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/Linkifier.java
@@ -0,0 +1,96 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+
+import com.google.common.base.Joiner;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.Lists;
+
+import java.util.List;
+import java.util.Map;
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
+
+import javax.servlet.http.HttpServletRequest;
+
+/** Linkifier for blocks of text such as commit message descriptions. */
+public class Linkifier {
+  private static final Pattern LINK_PATTERN;
+
+  static {
+    // HTTP URL regex adapted from com.google.gwtexpui.safehtml.client.SafeHtml.
+    String part = "[a-zA-Z0-9$_.+!*',%;:@=?#/~<>-]";
+    String httpUrl = "https?://" +
+        part + "{2,}" +
+        "(?:[(]" + part + "*" + "[)])*" +
+        part + "*";
+    String changeId = "\\bI[0-9a-f]{8,40}\\b";
+    LINK_PATTERN = Pattern.compile(Joiner.on("|").join(
+        "(" + httpUrl + ")",
+        "(" + changeId + ")"));
+  }
+
+  private final GitilesUrls urls;
+
+  public Linkifier(GitilesUrls urls) {
+    this.urls = checkNotNull(urls, "urls");
+  }
+
+  public List<Map<String, String>> linkify(HttpServletRequest req, String message) {
+    String baseGerritUrl = urls.getBaseGerritUrl(req);
+    List<Map<String, String>> parsed = Lists.newArrayList();
+    Matcher m = LINK_PATTERN.matcher(message);
+    int last = 0;
+    while (m.find()) {
+      addText(parsed, message.substring(last, m.start()));
+      if (m.group(1) != null) {
+        // Bare URL.
+        parsed.add(link(m.group(1), m.group(1)));
+      } else if (m.group(2) != null) {
+        if (baseGerritUrl != null) {
+          // Gerrit Change-Id.
+          parsed.add(link(m.group(2), baseGerritUrl + "#/q/" + m.group(2) + ",n,z"));
+        } else {
+          addText(parsed, m.group(2));
+        }
+      }
+      last = m.end();
+    }
+    addText(parsed, message.substring(last));
+    return parsed;
+  }
+
+  private static Map<String, String> link(String text, String url) {
+    return ImmutableMap.of("text", text, "url", url);
+  }
+
+  private static void addText(List<Map<String, String>> parts, String text) {
+    if (text.isEmpty()) {
+      return;
+    }
+    if (parts.isEmpty()) {
+      parts.add(ImmutableMap.of("text", text));
+    } else {
+      Map<String, String> old = parts.get(parts.size() - 1);
+      if (!old.containsKey("url")) {
+        parts.set(parts.size() - 1, ImmutableMap.of("text", old.get("text") + text));
+      } else {
+        parts.add(ImmutableMap.of("text", text));
+      }
+    }
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/LogServlet.java b/gitiles-servlet/src/main/java/com/google/gitiles/LogServlet.java
new file mode 100644
index 0000000..07977f7
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/LogServlet.java
@@ -0,0 +1,193 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkArgument;
+import static com.google.common.base.Preconditions.checkNotNull;
+import static javax.servlet.http.HttpServletResponse.SC_INTERNAL_SERVER_ERROR;
+import static javax.servlet.http.HttpServletResponse.SC_NOT_FOUND;
+
+import com.google.common.base.Optional;
+import com.google.common.base.Strings;
+import com.google.common.collect.Iterables;
+import com.google.common.collect.ListMultimap;
+import com.google.common.collect.Lists;
+import com.google.common.collect.Maps;
+import com.google.gitiles.CommitSoyData.KeySet;
+
+import org.eclipse.jgit.errors.IncorrectObjectTypeException;
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.errors.RevWalkException;
+import org.eclipse.jgit.http.server.ServletUtils;
+import org.eclipse.jgit.lib.AbbreviatedObjectId;
+import org.eclipse.jgit.lib.AnyObjectId;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.lib.ObjectReader;
+import org.eclipse.jgit.lib.Ref;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.revwalk.FollowFilter;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevObject;
+import org.eclipse.jgit.revwalk.RevTag;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.io.IOException;
+import java.util.Collection;
+import java.util.List;
+import java.util.Map;
+import java.util.Set;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Serves an HTML page with a shortlog for commits and paths. */
+public class LogServlet extends BaseServlet {
+  private static final Logger log = LoggerFactory.getLogger(LogServlet.class);
+
+  private static final String START_PARAM = "s";
+
+  private final Linkifier linkifier;
+  private final int limit;
+
+  public LogServlet(Renderer renderer, Linkifier linkifier) {
+    this(renderer, linkifier, 100);
+  }
+
+  private LogServlet(Renderer renderer, Linkifier linkifier, int limit) {
+    super(renderer);
+    this.linkifier = checkNotNull(linkifier, "linkifier");
+    checkArgument(limit >= 0, "limit must be positive: %s", limit);
+    this.limit = limit;
+  }
+
+  @Override
+  protected void doGet(HttpServletRequest req, HttpServletResponse res) throws IOException {
+    GitilesView view = ViewFilter.getView(req);
+    Repository repo = ServletUtils.getRepository(req);
+    RevWalk walk = null;
+    try {
+      try {
+        walk = newWalk(repo, view);
+      } catch (IncorrectObjectTypeException e) {
+        res.setStatus(SC_NOT_FOUND);
+        return;
+      }
+
+      Optional<ObjectId> start = getStart(view.getParameters(), walk.getObjectReader());
+      if (start == null) {
+        res.setStatus(SC_NOT_FOUND);
+        return;
+      }
+
+      Map<String, Object> data = Maps.newHashMapWithExpectedSize(5);
+
+      if (!view.getRevision().nameIsId()) {
+        List<Map<String, Object>> tags = Lists.newArrayListWithExpectedSize(1);
+        for (RevObject o : RevisionServlet.listObjects(walk, view.getRevision().getId())) {
+          if (o instanceof RevTag) {
+            tags.add(new TagSoyData(linkifier, req).toSoyData((RevTag) o));
+          }
+        }
+        if (!tags.isEmpty()) {
+          data.put("tags", tags);
+        }
+      }
+
+      Paginator paginator = new Paginator(walk, limit, start.orNull());
+      Map<AnyObjectId, Set<Ref>> refsById = repo.getAllRefsByPeeledObjectId();
+      List<Map<String, Object>> entries = Lists.newArrayListWithCapacity(limit);
+      for (RevCommit c : paginator) {
+        entries.add(new CommitSoyData(null, req, repo, walk, view, refsById)
+            .toSoyData(c, KeySet.SHORTLOG));
+      }
+
+      String title = "Log - ";
+      if (view.getOldRevision() != Revision.NULL) {
+        title += view.getRevisionRange();
+      } else {
+        title += view.getRevision().getName();
+      }
+
+      data.put("title", title);
+      data.put("entries", entries);
+      ObjectId next = paginator.getNextStart();
+      if (next != null) {
+        data.put("nextUrl", copyAndCanonicalize(view)
+            .replaceParam(START_PARAM, next.name())
+            .toUrl());
+      }
+      ObjectId prev = paginator.getPreviousStart();
+      if (prev != null) {
+        GitilesView.Builder prevView = copyAndCanonicalize(view);
+        if (!prevView.getRevision().getId().equals(prev)) {
+          prevView.replaceParam(START_PARAM, prev.name());
+        }
+        data.put("previousUrl", prevView.toUrl());
+      }
+
+      render(req, res, "gitiles.logDetail", data);
+    } catch (RevWalkException e) {
+      log.warn("Error in rev walk", e);
+      res.setStatus(SC_INTERNAL_SERVER_ERROR);
+      return;
+    } finally {
+      if (walk != null) {
+        walk.release();
+      }
+    }
+  }
+
+  private static GitilesView.Builder copyAndCanonicalize(GitilesView view) {
+    // Canonicalize the view by using full SHAs.
+    GitilesView.Builder copy = GitilesView.log().copyFrom(view)
+        .setRevision(view.getRevision());
+    if (view.getOldRevision() != Revision.NULL) {
+      copy.setOldRevision(view.getOldRevision());
+    }
+    return copy;
+  }
+
+  private static Optional<ObjectId> getStart(ListMultimap<String, String> params,
+      ObjectReader reader) throws IOException {
+    List<String> values = params.get(START_PARAM);
+    switch (values.size()) {
+      case 0:
+        return Optional.absent();
+      case 1:
+        Collection<ObjectId> ids = reader.resolve(AbbreviatedObjectId.fromString(values.get(0)));
+        if (ids.size() != 1) {
+          return null;
+        }
+        return Optional.of(Iterables.getOnlyElement(ids));
+      default:
+        return null;
+    }
+  }
+
+  private static RevWalk newWalk(Repository repo, GitilesView view)
+      throws MissingObjectException, IncorrectObjectTypeException, IOException {
+    RevWalk walk = new RevWalk(repo);
+    walk.markStart(walk.parseCommit(view.getRevision().getId()));
+    if (view.getOldRevision() != Revision.NULL) {
+      walk.markUninteresting(walk.parseCommit(view.getOldRevision().getId()));
+    }
+    if (!Strings.isNullOrEmpty(view.getTreePath())) {
+      walk.setTreeFilter(FollowFilter.create(view.getTreePath()));
+    }
+    return walk;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/Paginator.java b/gitiles-servlet/src/main/java/com/google/gitiles/Paginator.java
new file mode 100644
index 0000000..0454556
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/Paginator.java
@@ -0,0 +1,176 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkArgument;
+import static com.google.common.base.Preconditions.checkNotNull;
+import static com.google.common.base.Preconditions.checkState;
+
+import org.eclipse.jgit.errors.IncorrectObjectTypeException;
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.errors.RevWalkException;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevWalk;
+
+import java.io.IOException;
+import java.util.ArrayDeque;
+import java.util.Deque;
+import java.util.Iterator;
+
+import javax.annotation.Nullable;
+
+/**
+ * Wrapper around {@link RevWalk} that paginates for Gitiles.
+ *
+ * A single page of a shortlog is defined by a revision range, such as "master"
+ * or "master..next", a page size, and a start commit, such as "c0ffee". The
+ * distance between the first commit in the walk ("next") and the first commit
+ * in the page may be arbitrarily long, but in order to present the commit list
+ * in a stable way, we must always start from the first commit in the walk. This
+ * is because there may be arbitrary merge commits between "c0ffee" and "next"
+ * that effectively insert arbitrary commits into the history starting from
+ * "c0ffee".
+ */
+class Paginator implements Iterable<RevCommit> {
+  private final RevWalk walk;
+  private final ObjectId start;
+  private final int limit;
+  private final Deque<ObjectId> prevBuffer;
+
+  private boolean done;
+  private int i;
+  private int n;
+  private int foundIndex;
+  private ObjectId nextStart;
+
+  /**
+   * @param walk revision walk.
+   * @param limit page size.
+   * @param start commit at which to start the walk, or null to start at the
+   *     beginning.
+   */
+  Paginator(RevWalk walk, int limit, @Nullable ObjectId start) {
+    this.walk = checkNotNull(walk, "walk");
+    this.start = start;
+    checkArgument(limit > 0, "limit must be positive: %s", limit);
+    this.limit = limit;
+    prevBuffer = new ArrayDeque<ObjectId>(start != null ? limit : 0);
+    i = -1;
+    foundIndex = -1;
+  }
+
+  /**
+   * Get the next element in this page of the walk.
+   *
+   * @return the next element, or null if the walk is finished.
+   *
+   * @throws MissingObjectException See {@link RevWalk#next()}.
+   * @throws IncorrectObjectTypeException See {@link RevWalk#next()}.
+   * @throws IOException See {@link RevWalk#next()}.
+   */
+  public RevCommit next() throws MissingObjectException, IncorrectObjectTypeException,
+      IOException {
+    RevCommit commit;
+    if (foundIndex < 0) {
+      while (true) {
+        commit = walk.next();
+        if (commit == null) {
+          done = true;
+          return null;
+        }
+        i++;
+        if (start == null || start.equals(commit)) {
+          foundIndex = i;
+          break;
+        }
+        if (prevBuffer.size() == limit) {
+          prevBuffer.remove();
+        }
+        prevBuffer.add(commit);
+      }
+    } else {
+      commit = walk.next();
+    }
+
+    if (++n == limit) {
+      done = true;
+    } else if (n == limit + 1 || commit == null) {
+      nextStart = commit;
+      done = true;
+      return null;
+    }
+    return commit;
+  }
+
+  /**
+   * @return the ID at the start of the page of results preceding this one, or
+   *     null if this is the first page.
+   */
+  public ObjectId getPreviousStart() {
+    checkState(done, "getPreviousStart() invalid before walk done");
+    return prevBuffer.pollFirst();
+  }
+
+  /**
+   * @return the ID at the start of the page of results after this one, or null
+   *     if this is the last page.
+   */
+  public ObjectId getNextStart() {
+    checkState(done, "getNextStart() invalid before walk done");
+    return nextStart;
+  }
+
+  /**
+   * @return an iterator over the commits in this walk.
+   * @throws RevWalkException if an error occurred, wrapping the checked
+   *     exception from {@link #next()}.
+   */
+  @Override
+  public Iterator<RevCommit> iterator() {
+    return new Iterator<RevCommit>() {
+      RevCommit next = nextUnchecked();
+
+      @Override
+      public boolean hasNext() {
+        return next != null;
+      }
+
+      @Override
+      public RevCommit next() {
+        RevCommit r = next;
+        next = nextUnchecked();
+        return r;
+      }
+
+      @Override
+      public void remove() {
+        throw new UnsupportedOperationException();
+      }
+    };
+  }
+
+  private RevCommit nextUnchecked() {
+    try {
+      return next();
+    } catch (MissingObjectException e) {
+      throw new RevWalkException(e);
+    } catch (IncorrectObjectTypeException e) {
+      throw new RevWalkException(e);
+    } catch (IOException e) {
+      throw new RevWalkException(e);
+    }
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/PathServlet.java b/gitiles-servlet/src/main/java/com/google/gitiles/PathServlet.java
new file mode 100644
index 0000000..d228202
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/PathServlet.java
@@ -0,0 +1,276 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.gitiles.TreeSoyData.resolveTargetUrl;
+import static javax.servlet.http.HttpServletResponse.SC_INTERNAL_SERVER_ERROR;
+import static javax.servlet.http.HttpServletResponse.SC_NOT_FOUND;
+import static org.eclipse.jgit.lib.Constants.OBJ_BLOB;
+import static org.eclipse.jgit.lib.Constants.OBJ_COMMIT;
+import static org.eclipse.jgit.lib.Constants.OBJ_TREE;
+
+import com.google.common.base.Joiner;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.Maps;
+
+import org.eclipse.jgit.errors.ConfigInvalidException;
+import org.eclipse.jgit.errors.LargeObjectException;
+import org.eclipse.jgit.http.server.ServletUtils;
+import org.eclipse.jgit.lib.FileMode;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.lib.ObjectLoader;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevObject;
+import org.eclipse.jgit.revwalk.RevTree;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.submodule.SubmoduleWalk;
+import org.eclipse.jgit.treewalk.TreeWalk;
+import org.eclipse.jgit.util.RawParseUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.io.IOException;
+import java.util.Map;
+import java.util.regex.Pattern;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Serves an HTML page with detailed information about a path within a tree. */
+// TODO(dborowitz): Handle non-UTF-8 names.
+public class PathServlet extends BaseServlet {
+  private static final Logger log = LoggerFactory.getLogger(PathServlet.class);
+
+  /**
+   * Submodule URLs where we know there is a web page if the user visits the
+   * repository URL verbatim in a web browser.
+   */
+  private static final Pattern VERBATIM_SUBMODULE_URL_PATTERN =
+      Pattern.compile("^(" + Joiner.on('|').join(
+          "https?://[^.]+.googlesource.com/.*",
+          "https?://[^.]+.googlecode.com/.*",
+          "https?://code.google.com/p/.*",
+          "https?://github.com/.*") + ")$", Pattern.CASE_INSENSITIVE);
+
+  static enum FileType {
+    TREE(FileMode.TREE),
+    SYMLINK(FileMode.SYMLINK),
+    REGULAR_FILE(FileMode.REGULAR_FILE),
+    EXECUTABLE_FILE(FileMode.EXECUTABLE_FILE),
+    GITLINK(FileMode.GITLINK);
+
+    private final FileMode mode;
+
+    private FileType(FileMode mode) {
+      this.mode = mode;
+    }
+
+    static FileType forEntry(TreeWalk tw) {
+      int mode = tw.getRawMode(0);
+      for (FileType type : values()) {
+        if (type.mode.equals(mode)) {
+          return type;
+        }
+      }
+      return null;
+    }
+  }
+
+  public PathServlet(Renderer renderer) {
+    super(renderer);
+  }
+
+  @Override
+  protected void doGet(HttpServletRequest req, HttpServletResponse res) throws IOException {
+    GitilesView view = ViewFilter.getView(req);
+    Repository repo = ServletUtils.getRepository(req);
+
+    RevWalk rw = new RevWalk(repo);
+    try {
+      RevObject obj = rw.peel(rw.parseAny(view.getRevision().getId()));
+      RevTree root;
+
+      switch (obj.getType()) {
+        case OBJ_COMMIT:
+          root = ((RevCommit) obj).getTree();
+          break;
+        case OBJ_TREE:
+          root = (RevTree) obj;
+          break;
+        default:
+          res.setStatus(SC_NOT_FOUND);
+          return;
+      }
+
+      TreeWalk tw;
+      FileType type;
+      String path = view.getTreePath();
+      if (path.isEmpty()) {
+        tw = new TreeWalk(rw.getObjectReader());
+        tw.addTree(root);
+        tw.setRecursive(false);
+        type = FileType.TREE;
+      } else {
+        tw = TreeWalk.forPath(rw.getObjectReader(), path, root);
+        if (tw == null) {
+          res.setStatus(SC_NOT_FOUND);
+          return;
+        }
+        type = FileType.forEntry(tw);
+        if (type == FileType.TREE) {
+          tw.enterSubtree();
+          tw.setRecursive(false);
+        }
+      }
+
+      switch (type) {
+        case TREE:
+          showTree(req, res, rw, tw, obj);
+          break;
+        case SYMLINK:
+          showSymlink(req, res, rw, tw);
+          break;
+        case REGULAR_FILE:
+        case EXECUTABLE_FILE:
+          showFile(req, res, rw, tw);
+          break;
+        case GITLINK:
+          showGitlink(req, res, rw, tw, root);
+          break;
+        default:
+          log.error("Bad file type: %s", type);
+          res.setStatus(SC_NOT_FOUND);
+          break;
+      }
+    } catch (LargeObjectException e) {
+      res.setStatus(SC_INTERNAL_SERVER_ERROR);
+    } finally {
+      rw.release();
+    }
+  }
+
+  private void showTree(HttpServletRequest req, HttpServletResponse res, RevWalk rw, TreeWalk tw,
+      ObjectId id) throws IOException {
+    GitilesView view = ViewFilter.getView(req);
+    // TODO(sop): Allow caching trees by SHA-1 when no S cookie is sent.
+    render(req, res, "gitiles.pathDetail", ImmutableMap.of(
+        "title", !view.getTreePath().isEmpty() ? view.getTreePath() : "/",
+        "type", FileType.TREE.toString(),
+        "data", new TreeSoyData(rw, view).toSoyData(id, tw)));
+  }
+
+  private void showFile(HttpServletRequest req, HttpServletResponse res, RevWalk rw, TreeWalk tw)
+      throws IOException {
+    GitilesView view = ViewFilter.getView(req);
+    // TODO(sop): Allow caching files by SHA-1 when no S cookie is sent.
+    render(req, res, "gitiles.pathDetail", ImmutableMap.of(
+        "title", ViewFilter.getView(req).getTreePath(),
+        "type", FileType.forEntry(tw).toString(),
+        "data", new BlobSoyData(rw, view).toSoyData(tw.getPathString(), tw.getObjectId(0))));
+  }
+
+  private void showSymlink(HttpServletRequest req, HttpServletResponse res, RevWalk rw,
+      TreeWalk tw) throws IOException {
+    GitilesView view = ViewFilter.getView(req);
+    ObjectId id = tw.getObjectId(0);
+    Map<String, Object> data = Maps.newHashMap();
+
+    ObjectLoader loader = rw.getObjectReader().open(id, OBJ_BLOB);
+    String target;
+    try {
+      target = RawParseUtils.decode(loader.getCachedBytes(TreeSoyData.MAX_SYMLINK_SIZE));
+    } catch (LargeObjectException.OutOfMemory e) {
+      throw e;
+    } catch (LargeObjectException e) {
+      data.put("sha", ObjectId.toString(id));
+      data.put("data", null);
+      data.put("size", Long.toString(loader.getSize()));
+      render(req, res, "gitiles.pathDetail", ImmutableMap.of(
+          "title", ViewFilter.getView(req).getTreePath(),
+          "type", FileType.REGULAR_FILE.toString(),
+          "data", data));
+      return;
+    }
+
+    String url = resolveTargetUrl(
+        GitilesView.path()
+            .copyFrom(view)
+            .setTreePath(dirname(view.getTreePath()))
+            .build(),
+        target);
+    data.put("title", view.getTreePath());
+    data.put("target", target);
+    if (url != null) {
+      data.put("targetUrl", url);
+    }
+
+    // TODO(sop): Allow caching files by SHA-1 when no S cookie is sent.
+    render(req, res, "gitiles.pathDetail", ImmutableMap.of(
+        "title", ViewFilter.getView(req).getTreePath(),
+        "type", FileType.SYMLINK.toString(),
+        "data", data));
+  }
+
+  private static String dirname(String path) {
+    while (path.charAt(path.length() - 1) == '/') {
+      path = path.substring(0, path.length() - 1);
+    }
+    int lastSlash = path.lastIndexOf('/');
+    if (lastSlash > 0) {
+      return path.substring(0, lastSlash - 1);
+    } else if (lastSlash == 0) {
+      return "/";
+    } else {
+      return ".";
+    }
+  }
+
+  private void showGitlink(HttpServletRequest req, HttpServletResponse res, RevWalk rw,
+      TreeWalk tw, RevTree root) throws IOException {
+    GitilesView view = ViewFilter.getView(req);
+    SubmoduleWalk sw = SubmoduleWalk.forPath(ServletUtils.getRepository(req), root,
+        view.getTreePath());
+
+    String remoteUrl;
+    try {
+      remoteUrl = sw.getRemoteUrl();
+    } catch (ConfigInvalidException e) {
+      throw new IOException(e);
+    } finally {
+      sw.release();
+    }
+
+    Map<String, Object> data = Maps.newHashMap();
+    data.put("sha", ObjectId.toString(tw.getObjectId(0)));
+    data.put("remoteUrl", remoteUrl);
+
+      // TODO(dborowitz): Guess when we can put commit SHAs in the URL.
+      String httpUrl = resolveHttpUrl(remoteUrl);
+      if (httpUrl != null) {
+        data.put("httpUrl", httpUrl);
+      }
+
+      // TODO(sop): Allow caching links by SHA-1 when no S cookie is sent.
+      render(req, res, "gitiles.pathDetail", ImmutableMap.of(
+          "title", view.getTreePath(),
+          "type", FileType.GITLINK.toString(),
+          "data", data));
+  }
+
+  private static String resolveHttpUrl(String remoteUrl) {
+    return VERBATIM_SUBMODULE_URL_PATTERN.matcher(remoteUrl).matches() ? remoteUrl : null;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/Renderer.java b/gitiles-servlet/src/main/java/com/google/gitiles/Renderer.java
new file mode 100644
index 0000000..42ce635
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/Renderer.java
@@ -0,0 +1,107 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+
+import com.google.common.base.Charsets;
+import com.google.common.base.Function;
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.Lists;
+import com.google.common.collect.Maps;
+import com.google.template.soy.tofu.SoyTofu;
+
+import java.io.File;
+import java.io.IOException;
+import java.net.MalformedURLException;
+import java.net.URL;
+import java.util.List;
+import java.util.Map;
+
+import javax.servlet.http.HttpServletResponse;
+
+/** Renderer for Soy templates used by Gitiles. */
+public abstract class Renderer {
+  private static final List<String> SOY_FILENAMES = ImmutableList.of(
+      "Common.soy",
+      "DiffDetail.soy",
+      "HostIndex.soy",
+      "LogDetail.soy",
+      "ObjectDetail.soy",
+      "PathDetail.soy",
+      "RevisionDetail.soy",
+      "RepositoryIndex.soy");
+
+  public static final Map<String, String> STATIC_URL_GLOBALS = ImmutableMap.of(
+      "gitiles.CSS_URL", "gitiles.css",
+      "gitiles.PRETTIFY_CSS_URL", "prettify/prettify.css",
+      "gitiles.PRETTIFY_JS_URL", "prettify/prettify.js");
+
+  protected static final URL toFileURL(String filename) {
+    if (filename == null) {
+      return null;
+    }
+    try {
+      return new File(filename).toURI().toURL();
+    } catch (MalformedURLException e) {
+      throw new IllegalArgumentException(e);
+    }
+  }
+
+  protected ImmutableList<URL> templates;
+  protected ImmutableMap<String, String> globals;
+
+  protected Renderer(Function<String, URL> resourceMapper, Map<String, String> globals,
+      String staticPrefix, URL customTemplates) {
+    checkNotNull(staticPrefix, "staticPrefix");
+    List<URL> allTemplates = Lists.newArrayListWithCapacity(SOY_FILENAMES.size() + 1);
+    for (String filename : SOY_FILENAMES) {
+      allTemplates.add(resourceMapper.apply(filename));
+    }
+    if (customTemplates != null) {
+      allTemplates.add(customTemplates);
+    } else {
+      allTemplates.add(resourceMapper.apply("DefaultCustomTemplates.soy"));
+    }
+    templates = ImmutableList.copyOf(allTemplates);
+
+    Map<String, String> allGlobals = Maps.newHashMap();
+    for (Map.Entry<String, String> e : STATIC_URL_GLOBALS.entrySet()) {
+      allGlobals.put(e.getKey(), staticPrefix + e.getValue());
+    }
+    allGlobals.putAll(globals);
+    this.globals = ImmutableMap.copyOf(allGlobals);
+  }
+
+  public void render(HttpServletResponse res, String templateName) throws IOException {
+    render(res, templateName, ImmutableMap.<String, Object> of());
+  }
+
+  public void render(HttpServletResponse res, String templateName, Map<String, ?> soyData)
+      throws IOException {
+    res.setContentType("text/html");
+    res.setCharacterEncoding("UTF-8");
+    byte[] data = newRenderer(templateName).setData(soyData).render().getBytes(Charsets.UTF_8);
+    res.setContentLength(data.length);
+    res.getOutputStream().write(data);
+  }
+
+  SoyTofu.Renderer newRenderer(String templateName) {
+    return getTofu().newRenderer(templateName);
+  }
+
+  protected abstract SoyTofu getTofu();
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/RepositoryDescription.java b/gitiles-servlet/src/main/java/com/google/gitiles/RepositoryDescription.java
new file mode 100644
index 0000000..06ee4da
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/RepositoryDescription.java
@@ -0,0 +1,25 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import java.util.Map;
+
+/** Describes a repository in the {@link HostIndexServlet} JSON output. */
+public class RepositoryDescription {
+  public String name;
+  public String cloneUrl;
+  public String description;
+  public Map<String, String> branches;
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/RepositoryIndexServlet.java b/gitiles-servlet/src/main/java/com/google/gitiles/RepositoryIndexServlet.java
new file mode 100644
index 0000000..325068a
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/RepositoryIndexServlet.java
@@ -0,0 +1,80 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+
+import com.google.common.annotations.VisibleForTesting;
+import com.google.common.base.Strings;
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.Lists;
+
+import org.eclipse.jgit.http.server.ServletUtils;
+import org.eclipse.jgit.lib.Constants;
+import org.eclipse.jgit.lib.Ref;
+import org.eclipse.jgit.lib.RefComparator;
+import org.eclipse.jgit.lib.RefDatabase;
+
+import java.io.IOException;
+import java.util.Collection;
+import java.util.List;
+import java.util.Map;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Serves the index page for a repository, if accessed directly by a browser. */
+public class RepositoryIndexServlet extends BaseServlet {
+  private final GitilesAccess.Factory accessFactory;
+
+  public RepositoryIndexServlet(Renderer renderer, GitilesAccess.Factory accessFactory) {
+    super(renderer);
+    this.accessFactory = checkNotNull(accessFactory, "accessFactory");
+  }
+
+  @Override
+  protected void doGet(HttpServletRequest req, HttpServletResponse res) throws IOException {
+    render(req, res, "gitiles.repositoryIndex", buildData(req));
+  }
+
+  @VisibleForTesting
+  Map<String, ?> buildData(HttpServletRequest req) throws IOException {
+    RepositoryDescription desc = accessFactory.forRequest(req).getRepositoryDescription();
+    return ImmutableMap.of(
+        "cloneUrl", desc.cloneUrl,
+        "description", Strings.nullToEmpty(desc.description),
+        "branches", getRefs(req, Constants.R_HEADS),
+        "tags", getRefs(req, Constants.R_TAGS));
+  }
+
+  private List<Map<String, String>> getRefs(HttpServletRequest req, String prefix)
+      throws IOException {
+    RefDatabase refdb = ServletUtils.getRepository(req).getRefDatabase();
+    String repoName = ViewFilter.getView(req).getRepositoryName();
+    Collection<Ref> refs = RefComparator.sort(refdb.getRefs(prefix).values());
+    List<Map<String, String>> result = Lists.newArrayListWithCapacity(refs.size());
+
+    for (Ref ref : refs) {
+      String name = ref.getName().substring(prefix.length());
+      boolean needPrefix = !ref.getName().equals(refdb.getRef(name).getName());
+      result.add(ImmutableMap.of(
+          "url", GitilesView.log().copyFrom(req).setRevision(
+              Revision.unpeeled(needPrefix ? ref.getName() : name, ref.getObjectId())).toUrl(),
+          "name", name));
+    }
+
+    return result;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/Revision.java b/gitiles-servlet/src/main/java/com/google/gitiles/Revision.java
new file mode 100644
index 0000000..05778e0
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/Revision.java
@@ -0,0 +1,138 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkArgument;
+import static org.eclipse.jgit.lib.Constants.OBJ_BAD;
+import static org.eclipse.jgit.lib.Constants.OBJ_TAG;
+
+import com.google.common.annotations.VisibleForTesting;
+import com.google.common.base.Objects;
+
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.lib.AbbreviatedObjectId;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.revwalk.RevObject;
+import org.eclipse.jgit.revwalk.RevWalk;
+
+import java.io.IOException;
+
+/**
+ * Object encapsulating a single revision as seen by Gitiles.
+ * <p>
+ * A single revision consists of a name, an ID, and a type. Name parsing is done
+ * once per request by {@link RevisionParser}.
+ */
+public class Revision {
+  /** Sentinel indicating a missing or empty revision. */
+  public static final Revision NULL = peeled("", ObjectId.zeroId(), OBJ_BAD);
+
+  /** Common default branch given to clients. */
+  public static final Revision HEAD = named("HEAD");
+
+  private final String name;
+  private final ObjectId id;
+  private final int type;
+  private final ObjectId peeledId;
+  private final int peeledType;
+
+  public static Revision peeled(String name, RevObject obj) {
+    return peeled(name, obj, obj.getType());
+  }
+
+  public static Revision unpeeled(String name, ObjectId id) {
+    return peeled(name, id, OBJ_BAD);
+  }
+
+  public static Revision named(String name) {
+    return peeled(name, null, OBJ_BAD);
+  }
+
+  public static Revision peel(String name, ObjectId id, RevWalk walk)
+      throws MissingObjectException, IOException {
+    RevObject obj = walk.parseAny(id);
+    RevObject peeled = walk.peel(obj);
+    return new Revision(name, obj, obj.getType(), peeled, peeled.getType());
+  }
+
+  private static Revision peeled(String name, ObjectId id, int type) {
+    checkArgument(type != OBJ_TAG, "expected non-tag for %s/%s", name, id);
+    return new Revision(name, id, type, id, type);
+  }
+
+  @VisibleForTesting
+  Revision(String name, ObjectId id, int type, ObjectId peeledId, int peeledType) {
+    this.name = name;
+    this.id = id;
+    this.type = type;
+    this.peeledId = peeledId;
+    this.peeledType = peeledType;
+  }
+
+  public String getName() {
+    return name;
+  }
+
+  public int getType() {
+    return type;
+  }
+
+  public ObjectId getId() {
+    return id;
+  }
+
+  public ObjectId getPeeledId() {
+    return peeledId;
+  }
+
+  public int getPeeledType() {
+    return peeledType;
+  }
+
+  public boolean nameIsId() {
+    return AbbreviatedObjectId.isId(name)
+        && (AbbreviatedObjectId.fromString(name).prefixCompare(id) == 0);
+  }
+
+  @Override
+  public boolean equals(Object o) {
+    if (o instanceof Revision) {
+      Revision r = (Revision) o;
+      return Objects.equal(name, r.name)
+          && Objects.equal(id, r.id)
+          && Objects.equal(type, r.type)
+          && Objects.equal(peeledId, r.peeledId)
+          && Objects.equal(peeledType, r.peeledType);
+    }
+    return false;
+  }
+
+  @Override
+  public int hashCode() {
+    return Objects.hashCode(name, id, type, peeledId, peeledType);
+  }
+
+  @Override
+  public String toString() {
+    return Objects.toStringHelper(this)
+        .omitNullValues()
+        .add("name", name)
+        .add("id", id)
+        .add("type", type)
+        .add("peeledId", peeledId)
+        .add("peeledType", peeledType)
+        .toString();
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/RevisionParser.java b/gitiles-servlet/src/main/java/com/google/gitiles/RevisionParser.java
new file mode 100644
index 0000000..3e0acfa
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/RevisionParser.java
@@ -0,0 +1,214 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+
+import com.google.common.annotations.VisibleForTesting;
+import com.google.common.base.CharMatcher;
+import com.google.common.base.Objects;
+import com.google.common.base.Splitter;
+
+import org.eclipse.jgit.errors.IncorrectObjectTypeException;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevWalk;
+
+import java.io.IOException;
+
+/** Object to parse revisions out of Gitiles paths. */
+class RevisionParser {
+  static final Splitter PATH_SPLITTER = Splitter.on('/');
+  private static final Splitter OPERATOR_SPLITTER = Splitter.on(CharMatcher.anyOf("^~"));
+
+  static class Result {
+    private final Revision revision;
+    private final Revision oldRevision;
+    private final int pathStart;
+
+    @VisibleForTesting
+    Result(Revision revision) {
+      this(revision, null, revision.getName().length());
+    }
+
+    @VisibleForTesting
+    Result(Revision revision, Revision oldRevision, int pathStart) {
+      this.revision = revision;
+      this.oldRevision = oldRevision;
+      this.pathStart = pathStart;
+    }
+
+    public Revision getRevision() {
+      return revision;
+    }
+
+    public Revision getOldRevision() {
+      return oldRevision;
+    }
+
+    @Override
+    public boolean equals(Object o) {
+      if (o instanceof Result) {
+        Result r = (Result) o;
+        return Objects.equal(revision, r.revision)
+            && Objects.equal(oldRevision, r.oldRevision)
+            && Objects.equal(pathStart, r.pathStart);
+      }
+      return false;
+    }
+
+    @Override
+    public int hashCode() {
+      return Objects.hashCode(revision, oldRevision, pathStart);
+    }
+
+    @Override
+    public String toString() {
+      return Objects.toStringHelper(this)
+          .omitNullValues()
+          .add("revision", revision)
+          .add("oldRevision", oldRevision)
+          .add("pathStart", pathStart)
+          .toString();
+    }
+
+    int getPathStart() {
+      return pathStart;
+    }
+  }
+
+  private final Repository repo;
+  private final GitilesAccess access;
+  private final VisibilityCache cache;
+
+  RevisionParser(Repository repo, GitilesAccess access, VisibilityCache cache) {
+    this.repo = checkNotNull(repo, "repo");
+    this.access = checkNotNull(access, "access");
+    this.cache = checkNotNull(cache, "cache");
+  }
+
+  Result parse(String path) throws IOException {
+    RevWalk walk = new RevWalk(repo);
+    try {
+      Revision oldRevision = null;
+
+      StringBuilder b = new StringBuilder();
+      boolean first = true;
+      for (String part : PATH_SPLITTER.split(path)) {
+        if (part.isEmpty()) {
+          return null; // No valid revision contains empty segments.
+        }
+        if (!first) {
+          b.append('/');
+        }
+
+        if (oldRevision == null) {
+          int dots = part.indexOf("..");
+          int firstParent = part.indexOf("^!");
+          if (dots == 0 || firstParent == 0) {
+            return null;
+          } else if (dots > 0) {
+            b.append(part.substring(0, dots));
+            String oldName = b.toString();
+            if (!isValidRevision(oldName)) {
+              return null;
+            } else {
+              ObjectId old = repo.resolve(oldName);
+              if (old == null) {
+                return null;
+              }
+              oldRevision = Revision.peel(oldName, old, walk);
+            }
+            part = part.substring(dots + 2);
+            b = new StringBuilder();
+          } else if (firstParent > 0) {
+            if (firstParent != part.length() - 2) {
+              return null;
+            }
+            b.append(part.substring(0, part.length() - 2));
+            String name = b.toString();
+            if (!isValidRevision(name)) {
+              return null;
+            }
+            ObjectId id = repo.resolve(name);
+            if (id == null) {
+              return null;
+            }
+            RevCommit c;
+            try {
+              c = walk.parseCommit(id);
+            } catch (IncorrectObjectTypeException e) {
+              return null; // Not a commit, ^! is invalid.
+            }
+            if (c.getParentCount() > 0) {
+              oldRevision = Revision.peeled(name + "^", c.getParent(0));
+            } else {
+              oldRevision = Revision.NULL;
+            }
+            Result result = new Result(Revision.peeled(name, c), oldRevision, name.length() + 2);
+            return isVisible(walk, result) ? result : null;
+          }
+        }
+        b.append(part);
+
+        String name = b.toString();
+        if (!isValidRevision(name)) {
+          return null;
+        }
+        ObjectId id = repo.resolve(name);
+        if (id != null) {
+          int pathStart;
+          if (oldRevision == null) {
+            pathStart = name.length(); // foo
+          } else {
+            // foo..bar (foo may be empty)
+            pathStart = oldRevision.getName().length() + 2 + name.length();
+          }
+          Result result = new Result(Revision.peel(name, id, walk), oldRevision, pathStart);
+          return isVisible(walk, result) ? result : null;
+        }
+        first = false;
+      }
+      return null;
+    } finally {
+      walk.release();
+    }
+  }
+
+  private static boolean isValidRevision(String revision) {
+    // Disallow some uncommon but valid revision expressions that either we
+    // don't support or we represent differently in our URLs.
+    return revision.indexOf(':') < 0
+        && revision.indexOf("^{") < 0
+        && revision.indexOf('@') < 0;
+  }
+
+  private boolean isVisible(RevWalk walk, Result result) throws IOException {
+    String maybeRef = OPERATOR_SPLITTER.split(result.getRevision().getName()).iterator().next();
+    if (repo.getRef(maybeRef) != null) {
+      // Name contains a visible ref; skip expensive reachability check.
+      return true;
+    }
+    if (!cache.isVisible(repo, walk, access, result.getRevision().getId())) {
+      return false;
+    }
+    if (result.getOldRevision() != null && result.getOldRevision() != Revision.NULL) {
+      return cache.isVisible(repo, walk, access, result.getOldRevision().getId());
+    } else {
+      return true;
+    }
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/RevisionServlet.java b/gitiles-servlet/src/main/java/com/google/gitiles/RevisionServlet.java
new file mode 100644
index 0000000..66a7086
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/RevisionServlet.java
@@ -0,0 +1,136 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+import static javax.servlet.http.HttpServletResponse.SC_NOT_FOUND;
+import static org.eclipse.jgit.lib.Constants.OBJ_BLOB;
+import static org.eclipse.jgit.lib.Constants.OBJ_COMMIT;
+import static org.eclipse.jgit.lib.Constants.OBJ_TAG;
+import static org.eclipse.jgit.lib.Constants.OBJ_TREE;
+
+import com.google.common.collect.ImmutableMap;
+import com.google.common.collect.Lists;
+import com.google.gitiles.CommitSoyData.KeySet;
+
+import org.eclipse.jgit.errors.IncorrectObjectTypeException;
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.http.server.ServletUtils;
+import org.eclipse.jgit.lib.Constants;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevObject;
+import org.eclipse.jgit.revwalk.RevTag;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.io.IOException;
+import java.util.List;
+import java.util.Map;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Serves an HTML page with detailed information about a ref. */
+public class RevisionServlet extends BaseServlet {
+  private static final Logger log = LoggerFactory.getLogger(RevisionServlet.class);
+
+  private final Linkifier linkifier;
+
+  public RevisionServlet(Renderer renderer, Linkifier linkifier) {
+    super(renderer);
+    this.linkifier = checkNotNull(linkifier, "linkifier");
+  }
+
+  @Override
+  protected void doGet(HttpServletRequest req, HttpServletResponse res) throws IOException {
+    GitilesView view = ViewFilter.getView(req);
+    Repository repo = ServletUtils.getRepository(req);
+
+    RevWalk walk = new RevWalk(repo);
+    try {
+      List<RevObject> objects = listObjects(walk, view.getRevision().getId());
+      List<Map<String, ?>> soyObjects = Lists.newArrayListWithCapacity(objects.size());
+      boolean hasBlob = false;
+
+      // TODO(sop): Allow caching commits by SHA-1 when no S cookie is sent.
+      for (RevObject obj : objects) {
+        try {
+          switch (obj.getType()) {
+            case OBJ_COMMIT:
+              soyObjects.add(ImmutableMap.of(
+                  "type", Constants.TYPE_COMMIT,
+                  "data", new CommitSoyData(linkifier, req, repo, walk, view)
+                      .toSoyData((RevCommit) obj, KeySet.DETAIL_DIFF_TREE)));
+              break;
+            case OBJ_TREE:
+              soyObjects.add(ImmutableMap.of(
+                  "type", Constants.TYPE_TREE,
+                  "data", new TreeSoyData(walk, view).toSoyData(obj)));
+              break;
+            case OBJ_BLOB:
+              soyObjects.add(ImmutableMap.of(
+                  "type", Constants.TYPE_BLOB,
+                  "data", new BlobSoyData(walk, view).toSoyData(obj)));
+              hasBlob = true;
+              break;
+            case OBJ_TAG:
+              soyObjects.add(ImmutableMap.of(
+                  "type", Constants.TYPE_TAG,
+                  "data", new TagSoyData(linkifier, req).toSoyData((RevTag) obj)));
+              break;
+            default:
+              log.warn("Bad object type for %s: %s", ObjectId.toString(obj.getId()), obj.getType());
+              res.setStatus(SC_NOT_FOUND);
+              return;
+          }
+        } catch (MissingObjectException e) {
+          log.warn("Missing object " + ObjectId.toString(obj.getId()), e);
+          res.setStatus(SC_NOT_FOUND);
+          return;
+        } catch (IncorrectObjectTypeException e) {
+          log.warn("Incorrect object type for " + ObjectId.toString(obj.getId()), e);
+          res.setStatus(SC_NOT_FOUND);
+          return;
+        }
+      }
+
+      render(req, res, "gitiles.revisionDetail", ImmutableMap.of(
+          "title", view.getRevision().getName(),
+          "objects", soyObjects,
+          "hasBlob", hasBlob));
+    } finally {
+      walk.release();
+    }
+  }
+
+  // TODO(dborowitz): Extract this.
+  static List<RevObject> listObjects(RevWalk walk, ObjectId id)
+      throws MissingObjectException, IOException {
+    List<RevObject> objects = Lists.newArrayListWithExpectedSize(1);
+    while (true) {
+      RevObject cur = walk.parseAny(id);
+      objects.add(cur);
+      if (cur.getType() == Constants.OBJ_TAG) {
+        id = ((RevTag) cur).getObject();
+      } else {
+        break;
+      }
+    }
+    return objects;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/TagSoyData.java b/gitiles-servlet/src/main/java/com/google/gitiles/TagSoyData.java
new file mode 100644
index 0000000..9ddf1e5
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/TagSoyData.java
@@ -0,0 +1,50 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import com.google.common.collect.Maps;
+
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.revwalk.RevTag;
+import org.eclipse.jgit.util.GitDateFormatter;
+import org.eclipse.jgit.util.GitDateFormatter.Format;
+
+import java.util.Map;
+
+import javax.servlet.http.HttpServletRequest;
+
+/** Soy data converter for git tags. */
+public class TagSoyData {
+  private final Linkifier linkifier;
+  private final HttpServletRequest req;
+  private final GitDateFormatter dateFormatter;
+
+  public TagSoyData(Linkifier linkifier, HttpServletRequest req) {
+    this.linkifier = linkifier;
+    this.req = req;
+    this.dateFormatter = new GitDateFormatter(Format.DEFAULT);
+  }
+
+  public Map<String, Object> toSoyData(RevTag tag) {
+    Map<String, Object> data = Maps.newHashMapWithExpectedSize(4);
+    data.put("sha", ObjectId.toString(tag));
+    if (tag.getTaggerIdent() != null) {
+      data.put("tagger", CommitSoyData.toSoyData(tag.getTaggerIdent(), dateFormatter));
+    }
+    data.put("object", ObjectId.toString(tag.getObject()));
+    data.put("message", linkifier.linkify(req, tag.getFullMessage()));
+    return data;
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/TreeSoyData.java b/gitiles-servlet/src/main/java/com/google/gitiles/TreeSoyData.java
new file mode 100644
index 0000000..6cbf541
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/TreeSoyData.java
@@ -0,0 +1,160 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.gitiles.RevisionParser.PATH_SPLITTER;
+import static org.eclipse.jgit.lib.Constants.OBJ_COMMIT;
+
+import com.google.common.annotations.VisibleForTesting;
+import com.google.common.base.Charsets;
+import com.google.common.collect.Lists;
+import com.google.common.collect.Maps;
+import com.google.common.io.Files;
+import com.google.gitiles.PathServlet.FileType;
+
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.treewalk.TreeWalk;
+
+import java.io.IOException;
+import java.util.List;
+import java.util.Map;
+import java.util.StringTokenizer;
+
+/** Soy data converter for git trees. */
+public class TreeSoyData {
+  /**
+   * Number of characters to display for a symlink target. Targets longer than
+   * this are abbreviated for display in a tree listing.
+   */
+  private static final int MAX_SYMLINK_TARGET_LENGTH = 72;
+
+  /**
+   * Maximum number of bytes to load from a blob that claims to be a symlink. If
+   * the blob is larger than this byte limit it will be displayed as a binary
+   * file instead of as a symlink.
+   */
+  static final int MAX_SYMLINK_SIZE = 16 << 10;
+
+  static String resolveTargetUrl(GitilesView view, String target) {
+    if (target.startsWith("/")) {
+      return null;
+    }
+
+    // simplifyPath() normalizes "a/../../" to "a", so manually check whether
+    // the path leads above the git root.
+    int depth = new StringTokenizer(view.getTreePath(), "/").countTokens();
+    for (String part : PATH_SPLITTER.split(target)) {
+      if (part.equals("..")) {
+        depth--;
+        if (depth < 0) {
+          return null;
+        }
+      } else if (!part.isEmpty() && !part.equals(".")) {
+        depth++;
+      }
+    }
+
+    String path = Files.simplifyPath(view.getTreePath() + "/" + target);
+    return GitilesView.path()
+        .copyFrom(view)
+        .setTreePath(!path.equals(".") ? path : "")
+        .toUrl();
+  }
+
+  @VisibleForTesting
+  static String getTargetDisplayName(String target) {
+    if (target.length() <= MAX_SYMLINK_TARGET_LENGTH) {
+      return target;
+    } else {
+      int lastSlash = target.lastIndexOf('/');
+      // TODO(dborowitz): Doesn't abbreviate a long last path component.
+      return lastSlash >= 0 ? "..." + target.substring(lastSlash) : target;
+    }
+  }
+
+  private final RevWalk rw;
+  private final GitilesView view;
+
+  public TreeSoyData(RevWalk rw, GitilesView view) {
+    this.rw = rw;
+    this.view = view;
+  }
+
+  public Map<String, Object> toSoyData(ObjectId treeId, TreeWalk tw) throws MissingObjectException,
+         IOException {
+    List<Object> entries = Lists.newArrayList();
+    GitilesView.Builder urlBuilder = GitilesView.path().copyFrom(view);
+    while (tw.next()) {
+      FileType type = FileType.forEntry(tw);
+      String name = tw.getNameString();
+
+      switch (view.getType()) {
+        case PATH:
+          urlBuilder.setTreePath(view.getTreePath() + "/" + name);
+          break;
+        case REVISION:
+          // Got here from a tag pointing at a tree.
+          urlBuilder.setTreePath(name);
+          break;
+        default:
+          throw new IllegalStateException(String.format(
+              "Cannot render TreeSoyData from %s view", view.getType()));
+      }
+
+      String url = urlBuilder.toUrl();
+      if (type == FileType.TREE) {
+        name += "/";
+        url += "/";
+      }
+      Map<String, String> entry = Maps.newHashMapWithExpectedSize(4);
+      entry.put("type", type.toString());
+      entry.put("name", name);
+      entry.put("url", url);
+      if (type == FileType.SYMLINK) {
+        String target = new String(
+            rw.getObjectReader().open(tw.getObjectId(0)).getCachedBytes(),
+            Charsets.UTF_8);
+        // TODO(dborowitz): Merge Shawn's changes before copying these methods
+        // in.
+        entry.put("targetName", getTargetDisplayName(target));
+        String targetUrl = resolveTargetUrl(view, target);
+        if (targetUrl != null) {
+          entry.put("targetUrl", targetUrl);
+        }
+      }
+      entries.add(entry);
+    }
+
+    Map<String, Object> data = Maps.newHashMapWithExpectedSize(3);
+    data.put("sha", treeId.name());
+    data.put("entries", entries);
+
+    if (view.getType() == GitilesView.Type.PATH
+        && view.getRevision().getPeeledType() == OBJ_COMMIT) {
+      data.put("logUrl", GitilesView.log().copyFrom(view).toUrl());
+    }
+
+    return data;
+  }
+
+  public Map<String, Object> toSoyData(ObjectId treeId) throws MissingObjectException, IOException {
+    TreeWalk tw = new TreeWalk(rw.getObjectReader());
+    tw.addTree(treeId);
+    tw.setRecursive(false);
+    return toSoyData(treeId, tw);
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/ViewFilter.java b/gitiles-servlet/src/main/java/com/google/gitiles/ViewFilter.java
new file mode 100644
index 0000000..8874b1d
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/ViewFilter.java
@@ -0,0 +1,160 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkArgument;
+import static com.google.common.base.Preconditions.checkNotNull;
+import static javax.servlet.http.HttpServletResponse.SC_NOT_FOUND;
+
+import org.eclipse.jgit.http.server.ServletUtils;
+import org.eclipse.jgit.http.server.glue.WrappedRequest;
+import org.eclipse.jgit.lib.Constants;
+
+import java.io.IOException;
+import java.util.Map;
+
+import javax.servlet.FilterChain;
+import javax.servlet.ServletException;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Filter to parse URLs and convert them to {@link GitilesView}s. */
+public class ViewFilter extends AbstractHttpFilter {
+  // TODO(dborowitz): Make this public in JGit (or implement getRegexGroup
+  // upstream).
+  private static final String REGEX_GROUPS_ATTRIBUTE =
+      "org.eclipse.jgit.http.server.glue.MetaServlet.serveRegex";
+
+  private static final String VIEW_ATTIRBUTE = ViewFilter.class.getName() + "/View";
+
+  private static final String CMD_AUTO = "+";
+  private static final String CMD_DIFF = "+diff";
+  private static final String CMD_LOG = "+log";
+  private static final String CMD_SHOW = "+show";
+
+  public static GitilesView getView(HttpServletRequest req) {
+    return (GitilesView) req.getAttribute(VIEW_ATTIRBUTE);
+  }
+
+  static String getRegexGroup(HttpServletRequest req, int groupId) {
+    WrappedRequest[] groups = (WrappedRequest[]) req.getAttribute(REGEX_GROUPS_ATTRIBUTE);
+    return checkNotNull(groups)[groupId].getPathInfo();
+  }
+
+  static void setView(HttpServletRequest req, GitilesView view) {
+    req.setAttribute(VIEW_ATTIRBUTE, view);
+  }
+
+  static String trimLeadingSlash(String str) {
+    checkArgument(str.startsWith("/"), "expected string starting with a slash: %s", str);
+    return str.substring(1);
+  }
+
+  private final GitilesUrls urls;
+  private final GitilesAccess.Factory accessFactory;
+  private final VisibilityCache visibilityCache;
+
+  public ViewFilter(GitilesAccess.Factory accessFactory, GitilesUrls urls,
+      VisibilityCache visibilityCache) {
+    this.urls = checkNotNull(urls, "urls");
+    this.accessFactory = checkNotNull(accessFactory, "accessFactory");
+    this.visibilityCache = checkNotNull(visibilityCache, "visibilityCache");
+  }
+
+  @Override
+  public void doFilter(HttpServletRequest req, HttpServletResponse res, FilterChain chain)
+      throws IOException, ServletException {
+    GitilesView.Builder view = parse(req);
+    if (view == null) {
+      res.setStatus(SC_NOT_FOUND);
+      return;
+    }
+    @SuppressWarnings("unchecked")
+    Map<String, String[]> params = req.getParameterMap();
+    view.setHostName(urls.getHostName(req))
+        .setServletPath(req.getContextPath() + req.getServletPath())
+        .putAllParams(params);
+    setView(req, view.build());
+    try {
+      chain.doFilter(req, res);
+    } finally {
+      req.removeAttribute(VIEW_ATTIRBUTE);
+    }
+  }
+
+  private GitilesView.Builder parse(HttpServletRequest req) throws IOException {
+    String repoName = trimLeadingSlash(getRegexGroup(req, 1));
+    String command = getRegexGroup(req, 2);
+    String path = getRegexGroup(req, 3);
+
+    // Non-path cases.
+    if (repoName.isEmpty()) {
+      return GitilesView.hostIndex();
+    } else if (command.isEmpty()) {
+      return GitilesView.repositoryIndex().setRepositoryName(repoName);
+    } else if (path.isEmpty()) {
+      return null; // Command but no path.
+    }
+
+    path = trimLeadingSlash(path);
+    RevisionParser revParser = new RevisionParser(
+        ServletUtils.getRepository(req), accessFactory.forRequest(req), visibilityCache);
+    RevisionParser.Result result = revParser.parse(path);
+    if (result == null) {
+      return null;
+    }
+    path = path.substring(result.getPathStart());
+
+    command = getCommand(command, result, path);
+    GitilesView.Builder view;
+    if (CMD_LOG.equals(command)) {
+      view = GitilesView.log().setTreePath(path);
+    } else if (CMD_SHOW.equals(command)) {
+      if (path.isEmpty()) {
+        view = GitilesView.revision();
+      } else {
+        view = GitilesView.path().setTreePath(path);
+      }
+    } else if (CMD_DIFF.equals(command)) {
+      view = GitilesView.diff().setTreePath(path);
+    } else {
+      return null; // Bad command.
+    }
+    if (result.getOldRevision() != null) { // May be NULL.
+      view.setOldRevision(result.getOldRevision());
+    }
+    view.setRepositoryName(repoName)
+        .setRevision(result.getRevision());
+    return view;
+  }
+
+  private String getCommand(String command, RevisionParser.Result result, String path) {
+    // Note: if you change the mapping for +, make sure to change
+    // GitilesView.toUrl() correspondingly.
+    if (!CMD_AUTO.equals(command)) {
+      return command;
+    } else if (result.getOldRevision() != null) {
+      return CMD_DIFF;
+    }
+    Revision rev = result.getRevision();
+    if (rev.getPeeledType() != Constants.OBJ_COMMIT
+        || !path.isEmpty()
+        || result.getRevision().nameIsId()) {
+      return CMD_SHOW;
+    } else {
+      return CMD_LOG;
+    }
+  }
+}
diff --git a/gitiles-servlet/src/main/java/com/google/gitiles/VisibilityCache.java b/gitiles-servlet/src/main/java/com/google/gitiles/VisibilityCache.java
new file mode 100644
index 0000000..22969d1
--- /dev/null
+++ b/gitiles-servlet/src/main/java/com/google/gitiles/VisibilityCache.java
@@ -0,0 +1,189 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+import static com.google.common.base.Predicates.not;
+import static com.google.common.collect.Collections2.filter;
+
+import com.google.common.base.Objects;
+import com.google.common.base.Predicate;
+import com.google.common.base.Throwables;
+import com.google.common.cache.Cache;
+import com.google.common.cache.CacheBuilder;
+
+import org.eclipse.jgit.errors.IncorrectObjectTypeException;
+import org.eclipse.jgit.errors.MissingObjectException;
+import org.eclipse.jgit.lib.Constants;
+import org.eclipse.jgit.lib.ObjectId;
+import org.eclipse.jgit.lib.Ref;
+import org.eclipse.jgit.lib.RefDatabase;
+import org.eclipse.jgit.lib.Repository;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevSort;
+import org.eclipse.jgit.revwalk.RevWalk;
+
+import java.io.IOException;
+import java.util.Collection;
+import java.util.concurrent.Callable;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.TimeUnit;
+
+/** Cache of per-user object visibility. */
+public class VisibilityCache {
+  private static class Key {
+    private final Object user;
+    private final String repositoryName;
+    private final ObjectId objectId;
+
+    private Key(Object user, String repositoryName, ObjectId objectId) {
+      this.user = checkNotNull(user, "user");
+      this.repositoryName = checkNotNull(repositoryName, "repositoryName");
+      this.objectId = checkNotNull(objectId, "objectId");
+    }
+
+    @Override
+    public boolean equals(Object o) {
+      if (o instanceof Key) {
+        Key k = (Key) o;
+        return Objects.equal(user, k.user)
+            && Objects.equal(repositoryName, k.repositoryName)
+            && Objects.equal(objectId, k.objectId);
+      }
+      return false;
+    }
+
+    @Override
+    public int hashCode() {
+      return Objects.hashCode(user, repositoryName, objectId);
+    }
+
+    @Override
+    public String toString() {
+      return Objects.toStringHelper(this)
+        .add("user", user)
+        .add("repositoryName", repositoryName)
+        .add("objectId", objectId)
+        .toString();
+    }
+  }
+
+  private final Cache<Key, Boolean> cache;
+  private final boolean topoSort;
+
+  public static CacheBuilder<Object, Object> newBuilder() {
+    return CacheBuilder.newBuilder()
+        .maximumSize(1 << 10)
+        .expireAfterWrite(30, TimeUnit.MINUTES);
+  }
+
+  public VisibilityCache(boolean topoSort) {
+    this(topoSort, newBuilder());
+  }
+
+  public VisibilityCache(boolean topoSort, CacheBuilder<Object, Object> builder) {
+    this.cache = builder.build();
+    this.topoSort = topoSort;
+  }
+
+  public Cache<?, Boolean> getCache() {
+    return cache;
+  }
+
+  boolean isVisible(final Repository repo, final RevWalk walk, GitilesAccess access,
+      final ObjectId id) throws IOException {
+    try {
+      return cache.get(
+          new Key(access.getUserKey(), access.getRepositoryName(), id),
+          new Callable<Boolean>() {
+            @Override
+            public Boolean call() throws IOException {
+              return isVisible(repo, walk, id);
+            }
+          });
+    } catch (ExecutionException e) {
+      Throwables.propagateIfInstanceOf(e.getCause(), IOException.class);
+      throw new IOException(e);
+    }
+  }
+
+  private boolean isVisible(Repository repo, RevWalk walk, ObjectId id) throws IOException {
+    RevCommit commit;
+    try {
+      commit = walk.parseCommit(id);
+    } catch (IncorrectObjectTypeException e) {
+      return false;
+    }
+
+    // If any reference directly points at the requested object, permit display.
+    // Common for displays of pending patch sets in Gerrit Code Review, or
+    // bookmarks to the commit a tag points at.
+    Collection<Ref> allRefs = repo.getRefDatabase().getRefs(RefDatabase.ALL).values();
+    for (Ref ref : allRefs) {
+      ref = repo.getRefDatabase().peel(ref);
+      if (id.equals(ref.getObjectId()) || id.equals(ref.getPeeledObjectId())) {
+        return true;
+      }
+    }
+
+    // Check heads first under the assumption that most requests are for refs
+    // close to a head. Tags tend to be much further back in history and just
+    // clutter up the priority queue in the common case.
+    return isReachableFrom(walk, commit, filter(allRefs, refStartsWith(Constants.R_HEADS)))
+        || isReachableFrom(walk, commit, filter(allRefs, refStartsWith(Constants.R_TAGS)))
+        || isReachableFrom(walk, commit, filter(allRefs, not(refStartsWith("refs/changes/"))));
+  }
+
+  private static Predicate<Ref> refStartsWith(final String prefix) {
+    return new Predicate<Ref>() {
+      @Override
+      public boolean apply(Ref ref) {
+        return ref.getName().startsWith(prefix);
+      }
+    };
+  }
+
+  private boolean isReachableFrom(RevWalk walk, RevCommit commit, Collection<Ref> refs)
+      throws IOException {
+    walk.reset();
+    if (topoSort) {
+      walk.sort(RevSort.TOPO);
+    }
+    walk.markStart(commit);
+    for (Ref ref : refs) {
+      if (ref.getPeeledObjectId() != null) {
+        markUninteresting(walk, ref.getPeeledObjectId());
+      } else {
+        markUninteresting(walk, ref.getObjectId());
+      }
+    }
+    // If the commit is reachable from any branch head, it will appear to be
+    // uninteresting to the RevWalk and no output will be produced.
+    return walk.next() == null;
+  }
+
+  private static void markUninteresting(RevWalk walk, ObjectId id) throws IOException {
+    if (id == null) {
+      return;
+    }
+    try {
+      walk.markUninteresting(walk.parseCommit(id));
+    } catch (IncorrectObjectTypeException e) {
+      // Do nothing, doesn't affect reachability.
+    } catch (MissingObjectException e) {
+      // Do nothing, doesn't affect reachability.
+    }
+  }
+}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/gitiles.css b/gitiles-servlet/src/main/resources/com/google/gitiles/static/gitiles.css
new file mode 100644
index 0000000..0c57519
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/gitiles.css
@@ -0,0 +1,319 @@
+/**
+ * Copyright 2012 Google Inc. All Rights Reserved.
+ *
+ * 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.
+ */
+
+/* Common styles and definitions. */
+
+h1 {
+  position: absolute;
+  top: 0;
+  white-space: nowrap;
+  margin-top: 5px;
+}
+.menu {
+  position: absolute;
+  top: 0;
+  right: 0;
+  font-size: 10pt;
+  white-space: nowrap;
+  text-align: right;
+  margin-top: 5px;
+  margin-right: 5px;
+}
+.menu .entry {
+  padding-right: 5px;
+  border-right: 1px solid black;
+  margin-right: 0;
+}
+h2 {
+  margin-top: 3em;
+}
+.breadcrumbs {
+  margin-top: 3em;
+  font-size: 150%;
+  border-bottom: #ddd solid 1px; /* BORDER */
+}
+table.list {
+  margin-top: 1em;
+  width: 90%;
+}
+table.list tr.no-hover:hover {
+  background: #fff;
+}
+table.list tr:hover, ol.list li:hover, pre.prettyprint li:hover {
+  background: #eee; /* HOVER_BACKGROUND */
+}
+table.list td {
+  white-space: nowrap;
+  padding-top: 0.25em;
+  padding-bottom: 0.25em;
+}
+.log-link {
+  margin-left: 0.5em
+}
+
+
+/* Styles for the host index page. */
+
+.instructions {
+  width: 45em;
+  margin-left: 1em;
+  margin-right: 1em;
+  border-top: 1px solid #555;
+  border-bottom: 1px solid #555;
+  color: #555;
+}
+.instructions pre {
+  display: block;
+  margin-left: 1em;
+  border-left: 2px solid #060;
+  padding-left: 1em;
+  white-space: nowrap;
+}
+.footer {
+  text-align: right;
+  background: #eee;
+  padding-right: 1em;
+  width: 90%;
+}
+.footer a {
+  color: black;
+  font-weight: bold;
+  font-size: 70%;
+}
+
+
+/* Styles for the repository index page. */
+
+.repository-description {
+  border-bottom: #ddd solid 1px; /* BORDER */
+  padding-bottom: 5px; /* VPADDING */
+}
+.repository-refs {
+  width: 450px;
+}
+.repository-branches {
+  float: left;
+  width: 200px;
+}
+.repository-tags {
+  float: right;
+  width: 250px;
+}
+.clone-line {
+  background-color: #e5ecf9; /* BOX_BACKGROUND */
+  border: none;
+  margin: 5px /* VPADDING */ 0 0 0;
+  padding: 5px 2em; /* PADDING */
+  font-size: 9pt;
+}
+
+
+/* Styles for the object detail templates. */
+
+.sha1 {
+  color: #666;
+  font-size: 9pt;
+}
+div.sha1 {
+  padding-top: 5px; /* VPADDING */
+}
+
+.git-commit, .git-tag {
+  font-size: 9pt;
+  border-bottom: #ddd solid 1px; /* BORDER */
+  padding: 5px 2em; /* PADDING */
+}
+.git-commit table, .git-tag table {
+  margin: 0;
+}
+.git-commit table th, .git-tag table th {
+  text-align: right;
+}
+pre.commit-message, pre.tag-message {
+  border-bottom: #ddd solid 1px; /* BORDER */
+  padding: 5px 2em; /* PADDING */
+  color: #000;
+  font-size: 9pt;
+  margin: 0;
+}
+
+ul.diff-tree {
+  font-size: 9pt;
+  list-style-type: none;
+  margin: 0;
+  padding: 5px 2em; /* PADDING */
+}
+ul.diff-tree .add {
+  color: #060;
+}
+ul.diff-tree .delete {
+  color: #600;
+}
+ul.diff-tree .rename, ul.diff-tree .copy {
+  color: #006;
+}
+span.diff-link, ul.diff-tree .add, ul.diff-tree .modify, ul.diff-tree .delete,
+    ul.diff-tree .rename, ul.diff-tree .copy {
+  margin-left: 0.5em;
+}
+.diff-summary {
+  font-size: 9pt;
+  font-style: italic;
+  padding: 5px 2em; /* PADDING */
+  border-bottom: #ddd solid 1px; /* BORDER */
+}
+
+ol.files {
+  list-style-type: none;
+  margin-left: 1em;
+  font-size: 10pt;
+  line-height: normal;
+}
+
+/* Tree icons are taken from the public domain Apache standard icons:
+ * http://www.apache.org/icons/ */
+ol.files li.git-tree{
+  /* small/folder.png */
+  list-style-image: url();
+}
+ol.files li.symlink{
+  /* small/forward.png */
+  list-style-image: url();
+}
+ol.files li.regular-file{
+  /* small/text.png */
+  list-style-image: url()
+}
+ol.files li.executable-file{
+  /* small/patch.png */
+  list-style-image: url();
+}
+ol.files li.gitlink{
+  /* small/continued.png */
+  list-style-image:url();
+}
+
+
+/* Styles for the path detail page. */
+
+.symlink-detail, .gitlink-detail {
+  margin-left: 1em;
+  color: #666;
+  font-style: italic;
+  font-size: 10pt;
+}
+
+
+/* Styles for the log detail page. */
+
+ol.shortlog {
+  list-style-type: none;
+  margin: 0;
+  padding: 5px 2em; /* PADDING */
+}
+ol.shortlog li {
+  border-bottom: #ddd solid 1px; /* BORDER */
+  padding-top: 2px;
+  padding-bottom: 2px;
+}
+ol.shortlog li.first {
+  border-top: #ddd solid 1px; /* BORDER */
+}
+ol.shortlog li:hover {
+  background: #eee; /* HOVER_BACKGROUND */
+}
+ol.shortlog .sha1 {
+  font-family: monospace;
+}
+.log-nav {
+  margin-top: 5px;
+  text-align: center;
+}
+.author {
+  padding-left: 3px;
+}
+.time {
+  font-size: 9pt; /* SHORTLOG_SMALL_FONT_SIZE */
+  font-style: italic;
+}
+.branch-label, .tag-label {
+  font-size: 9pt; /* SHORTLOG_SMALL_FONT_SIZE */
+  margin-left: 3px;
+}
+a.branch-label {
+  color: #dd4b39;
+}
+a.tag-label {
+  color: #009933;
+}
+
+
+/* Styles for the diff detail template. */
+
+.diff-header {
+}
+.diff-git {
+  color: #444;
+  font-weight: bold;
+}
+a.diff-git:hover {
+  text-decoration: none;
+}
+.diff-header, .diff-unified {
+  color: #000;
+  font-size: 9pt;
+  margin: 0;
+  padding-left: 2em;
+}
+.diff-unified {
+  border-bottom: #ddd solid 1px; /* BORDER */
+}
+.diff-unified .h {
+  color: darkblue;
+}
+.diff-unified .d {
+  color: darkred;
+}
+.diff-unified .i {
+  color: darkgreen;
+}
+
+
+/* Override some styles from the default prettify.css. */
+
+/* Line numbers on all lines. */
+li.L0, li.L1, li.L2, li.L3, li.L4, li.L5, li.L6, li.L7, li.L8, li.L9 {
+  list-style-type: decimal;
+}
+
+/* Disable alternating line background color. */
+li.L0, li.L1, li.L2, li.L3, li.L4, li.L5, li.L6, li.L7, li.L8, li.L9 {
+  background: #fff;
+}
+
+pre.git-blob {
+  border-top: #ddd solid 1px; /* BORDER */
+  border-bottom: #ddd solid 1px; /* BORDER */
+  border-left: none;
+  border-right: none;
+  padding-left: 1em;
+  padding-bottom: 5px;
+  font-family: monospace;
+  font-size: 8pt;
+}
+pre.prettyprint ol {
+  color: grey;
+}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/COPYING b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/COPYING
new file mode 100644
index 0000000..37a41c0
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/COPYING
@@ -0,0 +1,214 @@
+google-code-prettify downloaded from:
+http://google-code-prettify.googlecode.com/files/prettify-small-1-Jun-2011.tar.bz2
+
+All files under the Apache License, with the following copyrights:
+prettify.js: Copyright (C) 2006 Google Inc.
+lang-apollo.js: Copyright (C) 2009 Onno Hommes.
+lang-clj.js: * @license Copyright (C) 2011 Google Inc.
+lang-css.js: Copyright (C) 2009 Google Inc.
+lang-go.js: Copyright (C) 2010 Google Inc.
+lang-hs.js: Copyright (C) 2009 Google Inc.
+lang-lisp.js: Copyright (C) 2008 Google Inc.
+lang-lua.js: Copyright (C) 2008 Google Inc.
+lang-ml.js: Copyright (C) 2008 Google Inc.
+lang-n.js: Copyright (C) 2011 Zimin A.V.
+lang-proto.js: Copyright (C) 2006 Google Inc.
+lang-scala.js: Copyright (C) 2010 Google Inc.
+lang-sql.js: Copyright (C) 2008 Google Inc.
+lang-tex.js: Copyright (C) 2011 Martin S.
+lang-vb.js: Copyright (C) 2009 Google Inc.
+lang-wiki.js: Copyright (C) 2009 Google Inc.
+lang-xq.js: Copyright (C) 2011 Patrick Wied
+
+===============================================================================
+
+                                 Apache License
+                           Version 2.0, January 2004
+                        http://www.apache.org/licenses/
+
+   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+   1. Definitions.
+
+      "License" shall mean the terms and conditions for use, reproduction,
+      and distribution as defined by Sections 1 through 9 of this document.
+
+      "Licensor" shall mean the copyright owner or entity authorized by
+      the copyright owner that is granting the License.
+
+      "Legal Entity" shall mean the union of the acting entity and all
+      other entities that control, are controlled by, or are under common
+      control with that entity. For the purposes of this definition,
+      "control" means (i) the power, direct or indirect, to cause the
+      direction or management of such entity, whether by contract or
+      otherwise, or (ii) ownership of fifty percent (50%) or more of the
+      outstanding shares, or (iii) beneficial ownership of such entity.
+
+      "You" (or "Your") shall mean an individual or Legal Entity
+      exercising permissions granted by this License.
+
+      "Source" form shall mean the preferred form for making modifications,
+      including but not limited to software source code, documentation
+      source, and configuration files.
+
+      "Object" form shall mean any form resulting from mechanical
+      transformation or translation of a Source form, including but
+      not limited to compiled object code, generated documentation,
+      and conversions to other media types.
+
+      "Work" shall mean the work of authorship, whether in Source or
+      Object form, made available under the License, as indicated by a
+      copyright notice that is included in or attached to the work
+      (an example is provided in the Appendix below).
+
+      "Derivative Works" shall mean any work, whether in Source or Object
+      form, that is based on (or derived from) the Work and for which the
+      editorial revisions, annotations, elaborations, or other modifications
+      represent, as a whole, an original work of authorship. For the purposes
+      of this License, Derivative Works shall not include works that remain
+      separable from, or merely link (or bind by name) to the interfaces of,
+      the Work and Derivative Works thereof.
+
+      "Contribution" shall mean any work of authorship, including
+      the original version of the Work and any modifications or additions
+      to that Work or Derivative Works thereof, that is intentionally
+      submitted to Licensor for inclusion in the Work by the copyright owner
+      or by an individual or Legal Entity authorized to submit on behalf of
+      the copyright owner. For the purposes of this definition, "submitted"
+      means any form of electronic, verbal, or written communication sent
+      to the Licensor or its representatives, including but not limited to
+      communication on electronic mailing lists, source code control systems,
+      and issue tracking systems that are managed by, or on behalf of, the
+      Licensor for the purpose of discussing and improving the Work, but
+      excluding communication that is conspicuously marked or otherwise
+      designated in writing by the copyright owner as "Not a Contribution."
+
+      "Contributor" shall mean Licensor and any individual or Legal Entity
+      on behalf of whom a Contribution has been received by Licensor and
+      subsequently incorporated within the Work.
+
+   2. Grant of Copyright License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      copyright license to reproduce, prepare Derivative Works of,
+      publicly display, publicly perform, sublicense, and distribute the
+      Work and such Derivative Works in Source or Object form.
+
+   3. Grant of Patent License. Subject to the terms and conditions of
+      this License, each Contributor hereby grants to You a perpetual,
+      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+      (except as stated in this section) patent license to make, have made,
+      use, offer to sell, sell, import, and otherwise transfer the Work,
+      where such license applies only to those patent claims licensable
+      by such Contributor that are necessarily infringed by their
+      Contribution(s) alone or by combination of their Contribution(s)
+      with the Work to which such Contribution(s) was submitted. If You
+      institute patent litigation against any entity (including a
+      cross-claim or counterclaim in a lawsuit) alleging that the Work
+      or a Contribution incorporated within the Work constitutes direct
+      or contributory patent infringement, then any patent licenses
+      granted to You under this License for that Work shall terminate
+      as of the date such litigation is filed.
+
+   4. Redistribution. You may reproduce and distribute copies of the
+      Work or Derivative Works thereof in any medium, with or without
+      modifications, and in Source or Object form, provided that You
+      meet the following conditions:
+
+      (a) You must give any other recipients of the Work or
+          Derivative Works a copy of this License; and
+
+      (b) You must cause any modified files to carry prominent notices
+          stating that You changed the files; and
+
+      (c) You must retain, in the Source form of any Derivative Works
+          that You distribute, all copyright, patent, trademark, and
+          attribution notices from the Source form of the Work,
+          excluding those notices that do not pertain to any part of
+          the Derivative Works; and
+
+      (d) If the Work includes a "NOTICE" text file as part of its
+          distribution, then any Derivative Works that You distribute must
+          include a readable copy of the attribution notices contained
+          within such NOTICE file, excluding those notices that do not
+          pertain to any part of the Derivative Works, in at least one
+          of the following places: within a NOTICE text file distributed
+          as part of the Derivative Works; within the Source form or
+          documentation, if provided along with the Derivative Works; or,
+          within a display generated by the Derivative Works, if and
+          wherever such third-party notices normally appear. The contents
+          of the NOTICE file are for informational purposes only and
+          do not modify the License. You may add Your own attribution
+          notices within Derivative Works that You distribute, alongside
+          or as an addendum to the NOTICE text from the Work, provided
+          that such additional attribution notices cannot be construed
+          as modifying the License.
+
+      You may add Your own copyright statement to Your modifications and
+      may provide additional or different license terms and conditions
+      for use, reproduction, or distribution of Your modifications, or
+      for any such Derivative Works as a whole, provided Your use,
+      reproduction, and distribution of the Work otherwise complies with
+      the conditions stated in this License.
+
+   5. Submission of Contributions. Unless You explicitly state otherwise,
+      any Contribution intentionally submitted for inclusion in the Work
+      by You to the Licensor shall be under the terms and conditions of
+      this License, without any additional terms or conditions.
+      Notwithstanding the above, nothing herein shall supersede or modify
+      the terms of any separate license agreement you may have executed
+      with Licensor regarding such Contributions.
+
+   6. Trademarks. This License does not grant permission to use the trade
+      names, trademarks, service marks, or product names of the Licensor,
+      except as required for reasonable and customary use in describing the
+      origin of the Work and reproducing the content of the NOTICE file.
+
+   7. Disclaimer of Warranty. Unless required by applicable law or
+      agreed to in writing, Licensor provides the Work (and each
+      Contributor provides its Contributions) on an "AS IS" BASIS,
+      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+      implied, including, without limitation, any warranties or conditions
+      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+      PARTICULAR PURPOSE. You are solely responsible for determining the
+      appropriateness of using or redistributing the Work and assume any
+      risks associated with Your exercise of permissions under this License.
+
+   8. Limitation of Liability. In no event and under no legal theory,
+      whether in tort (including negligence), contract, or otherwise,
+      unless required by applicable law (such as deliberate and grossly
+      negligent acts) or agreed to in writing, shall any Contributor be
+      liable to You for damages, including any direct, indirect, special,
+      incidental, or consequential damages of any character arising as a
+      result of this License or out of the use or inability to use the
+      Work (including but not limited to damages for loss of goodwill,
+      work stoppage, computer failure or malfunction, or any and all
+      other commercial damages or losses), even if such Contributor
+      has been advised of the possibility of such damages.
+
+   9. Accepting Warranty or Additional Liability. While redistributing
+      the Work or Derivative Works thereof, You may choose to offer,
+      and charge a fee for, acceptance of support, warranty, indemnity,
+      or other liability obligations and/or rights consistent with this
+      License. However, in accepting such obligations, You may act only
+      on Your own behalf and on Your sole responsibility, not on behalf
+      of any other Contributor, and only if You agree to indemnify,
+      defend, and hold each Contributor harmless for any liability
+      incurred by, or claims asserted against, such Contributor by reason
+      of your accepting any such warranty or additional liability.
+
+   END OF TERMS AND CONDITIONS
+
+   Copyright 2011 Mike Samuel et al
+
+   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.
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-apollo.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-apollo.js
new file mode 100644
index 0000000..7098baf
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-apollo.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["com",/^#[^\n\r]*/,null,"#"],["pln",/^[\t\n\r \xa0]+/,null,"\t\n\r Â\xa0"],["str",/^"(?:[^"\\]|\\[\S\s])*(?:"|$)/,null,'"']],[["kwd",/^(?:ADS|AD|AUG|BZF|BZMF|CAE|CAF|CA|CCS|COM|CS|DAS|DCA|DCOM|DCS|DDOUBL|DIM|DOUBLE|DTCB|DTCF|DV|DXCH|EDRUPT|EXTEND|INCR|INDEX|NDX|INHINT|LXCH|MASK|MSK|MP|MSU|NOOP|OVSK|QXCH|RAND|READ|RELINT|RESUME|RETURN|ROR|RXOR|SQUARE|SU|TCR|TCAA|OVSK|TCF|TC|TS|WAND|WOR|WRITE|XCH|XLQ|XXALQ|ZL|ZQ|ADD|ADZ|SUB|SUZ|MPY|MPR|MPZ|DVP|COM|ABS|CLA|CLZ|LDQ|STO|STQ|ALS|LLS|LRS|TRA|TSQ|TMI|TOV|AXT|TIX|DLY|INP|OUT)\s/,
+null],["typ",/^(?:-?GENADR|=MINUS|2BCADR|VN|BOF|MM|-?2CADR|-?[1-6]DNADR|ADRES|BBCON|[ES]?BANK=?|BLOCK|BNKSUM|E?CADR|COUNT\*?|2?DEC\*?|-?DNCHAN|-?DNPTR|EQUALS|ERASE|MEMORY|2?OCT|REMADR|SETLOC|SUBRO|ORG|BSS|BES|SYN|EQU|DEFINE|END)\s/,null],["lit",/^'(?:-*(?:\w|\\[!-~])(?:[\w-]*|\\[!-~])[!=?]?)?/],["pln",/^-*(?:[!-z]|\\[!-~])(?:[\w-]*|\\[!-~])[!=?]?/],["pun",/^[^\w\t\n\r "'-);\\\xa0]+/]]),["apollo","agc","aea"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-clj.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-clj.js
new file mode 100644
index 0000000..542a220
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-clj.js
@@ -0,0 +1,18 @@
+/*
+ Copyright (C) 2011 Google Inc.
+
+ 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.
+*/
+var a=null;
+PR.registerLangHandler(PR.createSimpleLexer([["opn",/^[([{]+/,a,"([{"],["clo",/^[)\]}]+/,a,")]}"],["com",/^;[^\n\r]*/,a,";"],["pln",/^[\t\n\r \xa0]+/,a,"\t\n\r \xa0"],["str",/^"(?:[^"\\]|\\[\S\s])*(?:"|$)/,a,'"']],[["kwd",/^(?:def|if|do|let|quote|var|fn|loop|recur|throw|try|monitor-enter|monitor-exit|defmacro|defn|defn-|macroexpand|macroexpand-1|for|doseq|dosync|dotimes|and|or|when|not|assert|doto|proxy|defstruct|first|rest|cons|defprotocol|deftype|defrecord|reify|defmulti|defmethod|meta|with-meta|ns|in-ns|create-ns|import|intern|refer|alias|namespace|resolve|ref|deref|refset|new|set!|memfn|to-array|into-array|aset|gen-class|reduce|map|filter|find|nil?|empty?|hash-map|hash-set|vec|vector|seq|flatten|reverse|assoc|dissoc|list|list?|disj|get|union|difference|intersection|extend|extend-type|extend-protocol|prn)\b/,a],
+["typ",/^:[\dA-Za-z-]+/]]),["clj"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-css.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-css.js
new file mode 100644
index 0000000..041e1f5
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-css.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\f\r ]+/,null," \t\r\n"]],[["str",/^"(?:[^\n\f\r"\\]|\\(?:\r\n?|\n|\f)|\\[\S\s])*"/,null],["str",/^'(?:[^\n\f\r'\\]|\\(?:\r\n?|\n|\f)|\\[\S\s])*'/,null],["lang-css-str",/^url\(([^"')]*)\)/i],["kwd",/^(?:url|rgb|!important|@import|@page|@media|@charset|inherit)(?=[^\w-]|$)/i,null],["lang-css-kw",/^(-?(?:[_a-z]|\\[\da-f]+ ?)(?:[\w-]|\\\\[\da-f]+ ?)*)\s*:/i],["com",/^\/\*[^*]*\*+(?:[^*/][^*]*\*+)*\//],["com",
+/^(?:<\!--|--\>)/],["lit",/^(?:\d+|\d*\.\d+)(?:%|[a-z]+)?/i],["lit",/^#[\da-f]{3,6}/i],["pln",/^-?(?:[_a-z]|\\[\da-f]+ ?)(?:[\w-]|\\\\[\da-f]+ ?)*/i],["pun",/^[^\s\w"']+/]]),["css"]);PR.registerLangHandler(PR.createSimpleLexer([],[["kwd",/^-?(?:[_a-z]|\\[\da-f]+ ?)(?:[\w-]|\\\\[\da-f]+ ?)*/i]]),["css-kw"]);PR.registerLangHandler(PR.createSimpleLexer([],[["str",/^[^"')]+/]]),["css-str"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-go.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-go.js
new file mode 100644
index 0000000..fc18dc0
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-go.js
@@ -0,0 +1 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\r \xa0]+/,null,"\t\n\r Â\xa0"],["pln",/^(?:"(?:[^"\\]|\\[\S\s])*(?:"|$)|'(?:[^'\\]|\\[\S\s])+(?:'|$)|`[^`]*(?:`|$))/,null,"\"'"]],[["com",/^(?:\/\/[^\n\r]*|\/\*[\S\s]*?\*\/)/],["pln",/^(?:[^"'/`]|\/(?![*/]))+/]]),["go"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-hs.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-hs.js
new file mode 100644
index 0000000..9d77b08
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-hs.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t-\r ]+/,null,"\t\n\r "],["str",/^"(?:[^\n\f\r"\\]|\\[\S\s])*(?:"|$)/,null,'"'],["str",/^'(?:[^\n\f\r'\\]|\\[^&])'?/,null,"'"],["lit",/^(?:0o[0-7]+|0x[\da-f]+|\d+(?:\.\d+)?(?:e[+-]?\d+)?)/i,null,"0123456789"]],[["com",/^(?:--+[^\n\f\r]*|{-(?:[^-]|-+[^}-])*-})/],["kwd",/^(?:case|class|data|default|deriving|do|else|if|import|in|infix|infixl|infixr|instance|let|module|newtype|of|then|type|where|_)(?=[^\d'A-Za-z]|$)/,
+null],["pln",/^(?:[A-Z][\w']*\.)*[A-Za-z][\w']*/],["pun",/^[^\d\t-\r "'A-Za-z]+/]]),["hs"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-lisp.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-lisp.js
new file mode 100644
index 0000000..02a30e8
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-lisp.js
@@ -0,0 +1,3 @@
+var a=null;
+PR.registerLangHandler(PR.createSimpleLexer([["opn",/^\(+/,a,"("],["clo",/^\)+/,a,")"],["com",/^;[^\n\r]*/,a,";"],["pln",/^[\t\n\r \xa0]+/,a,"\t\n\r \xa0"],["str",/^"(?:[^"\\]|\\[\S\s])*(?:"|$)/,a,'"']],[["kwd",/^(?:block|c[ad]+r|catch|con[ds]|def(?:ine|un)|do|eq|eql|equal|equalp|eval-when|flet|format|go|if|labels|lambda|let|load-time-value|locally|macrolet|multiple-value-call|nil|progn|progv|quote|require|return-from|setq|symbol-macrolet|t|tagbody|the|throw|unwind)\b/,a],
+["lit",/^[+-]?(?:[#0]x[\da-f]+|\d+\/\d+|(?:\.\d+|\d+(?:\.\d*)?)(?:[de][+-]?\d+)?)/i],["lit",/^'(?:-*(?:\w|\\[!-~])(?:[\w-]*|\\[!-~])[!=?]?)?/],["pln",/^-*(?:[_a-z]|\\[!-~])(?:[\w-]*|\\[!-~])[!=?]?/i],["pun",/^[^\w\t\n\r "'-);\\\xa0]+/]]),["cl","el","lisp","scm"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-lua.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-lua.js
new file mode 100644
index 0000000..e83a3c4
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-lua.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\r \xa0]+/,null,"\t\n\r Â\xa0"],["str",/^(?:"(?:[^"\\]|\\[\S\s])*(?:"|$)|'(?:[^'\\]|\\[\S\s])*(?:'|$))/,null,"\"'"]],[["com",/^--(?:\[(=*)\[[\S\s]*?(?:]\1]|$)|[^\n\r]*)/],["str",/^\[(=*)\[[\S\s]*?(?:]\1]|$)/],["kwd",/^(?:and|break|do|else|elseif|end|false|for|function|if|in|local|nil|not|or|repeat|return|then|true|until|while)\b/,null],["lit",/^[+-]?(?:0x[\da-f]+|(?:\.\d+|\d+(?:\.\d*)?)(?:e[+-]?\d+)?)/i],
+["pln",/^[_a-z]\w*/i],["pun",/^[^\w\t\n\r \xa0][^\w\t\n\r "'+=\xa0-]*/]]),["lua"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-ml.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-ml.js
new file mode 100644
index 0000000..6df02d7
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-ml.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\r \xa0]+/,null,"\t\n\r Â\xa0"],["com",/^#(?:if[\t\n\r \xa0]+(?:[$_a-z][\w']*|``[^\t\n\r`]*(?:``|$))|else|endif|light)/i,null,"#"],["str",/^(?:"(?:[^"\\]|\\[\S\s])*(?:"|$)|'(?:[^'\\]|\\[\S\s])(?:'|$))/,null,"\"'"]],[["com",/^(?:\/\/[^\n\r]*|\(\*[\S\s]*?\*\))/],["kwd",/^(?:abstract|and|as|assert|begin|class|default|delegate|do|done|downcast|downto|elif|else|end|exception|extern|false|finally|for|fun|function|if|in|inherit|inline|interface|internal|lazy|let|match|member|module|mutable|namespace|new|null|of|open|or|override|private|public|rec|return|static|struct|then|to|true|try|type|upcast|use|val|void|when|while|with|yield|asr|land|lor|lsl|lsr|lxor|mod|sig|atomic|break|checked|component|const|constraint|constructor|continue|eager|event|external|fixed|functor|global|include|method|mixin|object|parallel|process|protected|pure|sealed|trait|virtual|volatile)\b/],
+["lit",/^[+-]?(?:0x[\da-f]+|(?:\.\d+|\d+(?:\.\d*)?)(?:e[+-]?\d+)?)/i],["pln",/^(?:[_a-z][\w']*[!#?]?|``[^\t\n\r`]*(?:``|$))/i],["pun",/^[^\w\t\n\r "'\xa0]+/]]),["fs","ml"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-n.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-n.js
new file mode 100644
index 0000000..6c2e85b
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-n.js
@@ -0,0 +1,4 @@
+var a=null;
+PR.registerLangHandler(PR.createSimpleLexer([["str",/^(?:'(?:[^\n\r'\\]|\\.)*'|"(?:[^\n\r"\\]|\\.)*(?:"|$))/,a,'"'],["com",/^#(?:(?:define|elif|else|endif|error|ifdef|include|ifndef|line|pragma|undef|warning)\b|[^\n\r]*)/,a,"#"],["pln",/^\s+/,a," \r\n\t\xa0"]],[["str",/^@"(?:[^"]|"")*(?:"|$)/,a],["str",/^<#[^#>]*(?:#>|$)/,a],["str",/^<(?:(?:(?:\.\.\/)*|\/?)(?:[\w-]+(?:\/[\w-]+)+)?[\w-]+\.h|[a-z]\w*)>/,a],["com",/^\/\/[^\n\r]*/,a],["com",/^\/\*[\S\s]*?(?:\*\/|$)/,
+a],["kwd",/^(?:abstract|and|as|base|catch|class|def|delegate|enum|event|extern|false|finally|fun|implements|interface|internal|is|macro|match|matches|module|mutable|namespace|new|null|out|override|params|partial|private|protected|public|ref|sealed|static|struct|syntax|this|throw|true|try|type|typeof|using|variant|virtual|volatile|when|where|with|assert|assert2|async|break|checked|continue|do|else|ensures|for|foreach|if|late|lock|new|nolate|otherwise|regexp|repeat|requires|return|surroundwith|unchecked|unless|using|while|yield)\b/,
+a],["typ",/^(?:array|bool|byte|char|decimal|double|float|int|list|long|object|sbyte|short|string|ulong|uint|ufloat|ulong|ushort|void)\b/,a],["lit",/^@[$_a-z][\w$@]*/i,a],["typ",/^@[A-Z]+[a-z][\w$@]*/,a],["pln",/^'?[$_a-z][\w$@]*/i,a],["lit",/^(?:0x[\da-f]+|(?:\d(?:_\d+)*\d*(?:\.\d*)?|\.\d\+)(?:e[+-]?\d+)?)[a-z]*/i,a,"0123456789"],["pun",/^.[^\s\w"-$'./@`]*/,a]]),["n","nemerle"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-proto.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-proto.js
new file mode 100644
index 0000000..f006ad8
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-proto.js
@@ -0,0 +1 @@
+PR.registerLangHandler(PR.sourceDecorator({keywords:"bytes,default,double,enum,extend,extensions,false,group,import,max,message,option,optional,package,repeated,required,returns,rpc,service,syntax,to,true",types:/^(bool|(double|s?fixed|[su]?int)(32|64)|float|string)\b/,cStyleComments:!0}),["proto"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-scala.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-scala.js
new file mode 100644
index 0000000..60d034d
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-scala.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\r \xa0]+/,null,"\t\n\r Â\xa0"],["str",/^"(?:""(?:""?(?!")|[^"\\]|\\.)*"{0,3}|(?:[^\n\r"\\]|\\.)*"?)/,null,'"'],["lit",/^`(?:[^\n\r\\`]|\\.)*`?/,null,"`"],["pun",/^[!#%&(--:-@[-^{-~]+/,null,"!#%&()*+,-:;<=>?@[\\]^{|}~"]],[["str",/^'(?:[^\n\r'\\]|\\(?:'|[^\n\r']+))'/],["lit",/^'[$A-Z_a-z][\w$]*(?![\w$'])/],["kwd",/^(?:abstract|case|catch|class|def|do|else|extends|final|finally|for|forSome|if|implicit|import|lazy|match|new|object|override|package|private|protected|requires|return|sealed|super|throw|trait|try|type|val|var|while|with|yield)\b/],
+["lit",/^(?:true|false|null|this)\b/],["lit",/^(?:0(?:[0-7]+|x[\da-f]+)l?|(?:0|[1-9]\d*)(?:(?:\.\d+)?(?:e[+-]?\d+)?f?|l?)|\\.\d+(?:e[+-]?\d+)?f?)/i],["typ",/^[$_]*[A-Z][\d$A-Z_]*[a-z][\w$]*/],["pln",/^[$A-Z_a-z][\w$]*/],["com",/^\/(?:\/.*|\*(?:\/|\**[^*/])*(?:\*+\/?)?)/],["pun",/^(?:\.+|\/)/]]),["scala"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-sql.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-sql.js
new file mode 100644
index 0000000..da705b0
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-sql.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\r \xa0]+/,null,"\t\n\r Â\xa0"],["str",/^(?:"(?:[^"\\]|\\.)*"|'(?:[^'\\]|\\.)*')/,null,"\"'"]],[["com",/^(?:--[^\n\r]*|\/\*[\S\s]*?(?:\*\/|$))/],["kwd",/^(?:add|all|alter|and|any|as|asc|authorization|backup|begin|between|break|browse|bulk|by|cascade|case|check|checkpoint|close|clustered|coalesce|collate|column|commit|compute|constraint|contains|containstable|continue|convert|create|cross|current|current_date|current_time|current_timestamp|current_user|cursor|database|dbcc|deallocate|declare|default|delete|deny|desc|disk|distinct|distributed|double|drop|dummy|dump|else|end|errlvl|escape|except|exec|execute|exists|exit|fetch|file|fillfactor|for|foreign|freetext|freetexttable|from|full|function|goto|grant|group|having|holdlock|identity|identitycol|identity_insert|if|in|index|inner|insert|intersect|into|is|join|key|kill|left|like|lineno|load|match|merge|national|nocheck|nonclustered|not|null|nullif|of|off|offsets|on|open|opendatasource|openquery|openrowset|openxml|option|or|order|outer|over|percent|plan|precision|primary|print|proc|procedure|public|raiserror|read|readtext|reconfigure|references|replication|restore|restrict|return|revoke|right|rollback|rowcount|rowguidcol|rule|save|schema|select|session_user|set|setuser|shutdown|some|statistics|system_user|table|textsize|then|to|top|tran|transaction|trigger|truncate|tsequal|union|unique|update|updatetext|use|user|using|values|varying|view|waitfor|when|where|while|with|writetext)(?=[^\w-]|$)/i,
+null],["lit",/^[+-]?(?:0x[\da-f]+|(?:\.\d+|\d+(?:\.\d*)?)(?:e[+-]?\d+)?)/i],["pln",/^[_a-z][\w-]*/i],["pun",/^[^\w\t\n\r "'\xa0][^\w\t\n\r "'+\xa0-]*/]]),["sql"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-tex.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-tex.js
new file mode 100644
index 0000000..ce96fbb
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-tex.js
@@ -0,0 +1 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\r \xa0]+/,null,"\t\n\r Â\xa0"],["com",/^%[^\n\r]*/,null,"%"]],[["kwd",/^\\[@-Za-z]+/],["kwd",/^\\./],["typ",/^[$&]/],["lit",/[+-]?(?:\.\d+|\d+(?:\.\d*)?)(cm|em|ex|in|pc|pt|bp|mm)/i],["pun",/^[()=[\]{}]+/]]),["latex","tex"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-vb.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-vb.js
new file mode 100644
index 0000000..07506b0
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-vb.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\r \xa0\u2028\u2029]+/,null,"\t\n\r Â\xa0

"],["str",/^(?:["\u201c\u201d](?:[^"\u201c\u201d]|["\u201c\u201d]{2})(?:["\u201c\u201d]c|$)|["\u201c\u201d](?:[^"\u201c\u201d]|["\u201c\u201d]{2})*(?:["\u201c\u201d]|$))/i,null,'"“”'],["com",/^['\u2018\u2019].*/,null,"'‘’"]],[["kwd",/^(?:addhandler|addressof|alias|and|andalso|ansi|as|assembly|auto|boolean|byref|byte|byval|call|case|catch|cbool|cbyte|cchar|cdate|cdbl|cdec|char|cint|class|clng|cobj|const|cshort|csng|cstr|ctype|date|decimal|declare|default|delegate|dim|directcast|do|double|each|else|elseif|end|endif|enum|erase|error|event|exit|finally|for|friend|function|get|gettype|gosub|goto|handles|if|implements|imports|in|inherits|integer|interface|is|let|lib|like|long|loop|me|mod|module|mustinherit|mustoverride|mybase|myclass|namespace|new|next|not|notinheritable|notoverridable|object|on|option|optional|or|orelse|overloads|overridable|overrides|paramarray|preserve|private|property|protected|public|raiseevent|readonly|redim|removehandler|resume|return|select|set|shadows|shared|short|single|static|step|stop|string|structure|sub|synclock|then|throw|to|try|typeof|unicode|until|variant|wend|when|while|with|withevents|writeonly|xor|endif|gosub|let|variant|wend)\b/i,
+null],["com",/^rem.*/i],["lit",/^(?:true\b|false\b|nothing\b|\d+(?:e[+-]?\d+[dfr]?|[dfilrs])?|(?:&h[\da-f]+|&o[0-7]+)[ils]?|\d*\.\d+(?:e[+-]?\d+)?[dfr]?|#\s+(?:\d+[/-]\d+[/-]\d+(?:\s+\d+:\d+(?::\d+)?(\s*(?:am|pm))?)?|\d+:\d+(?::\d+)?(\s*(?:am|pm))?)\s+#)/i],["pln",/^(?:(?:[a-z]|_\w)\w*|\[(?:[a-z]|_\w)\w*])/i],["pun",/^[^\w\t\n\r "'[\]\xa0\u2018\u2019\u201c\u201d\u2028\u2029]+/],["pun",/^(?:\[|])/]]),["vb","vbs"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-vhdl.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-vhdl.js
new file mode 100644
index 0000000..128b5b6
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-vhdl.js
@@ -0,0 +1,3 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\t\n\r \xa0]+/,null,"\t\n\r Â\xa0"]],[["str",/^(?:[box]?"(?:[^"]|"")*"|'.')/i],["com",/^--[^\n\r]*/],["kwd",/^(?:abs|access|after|alias|all|and|architecture|array|assert|attribute|begin|block|body|buffer|bus|case|component|configuration|constant|disconnect|downto|else|elsif|end|entity|exit|file|for|function|generate|generic|group|guarded|if|impure|in|inertial|inout|is|label|library|linkage|literal|loop|map|mod|nand|new|next|nor|not|null|of|on|open|or|others|out|package|port|postponed|procedure|process|pure|range|record|register|reject|rem|report|return|rol|ror|select|severity|shared|signal|sla|sll|sra|srl|subtype|then|to|transport|type|unaffected|units|until|use|variable|wait|when|while|with|xnor|xor)(?=[^\w-]|$)/i,
+null],["typ",/^(?:bit|bit_vector|character|boolean|integer|real|time|string|severity_level|positive|natural|signed|unsigned|line|text|std_u?logic(?:_vector)?)(?=[^\w-]|$)/i,null],["typ",/^'(?:active|ascending|base|delayed|driving|driving_value|event|high|image|instance_name|last_active|last_event|last_value|left|leftof|length|low|path_name|pos|pred|quiet|range|reverse_range|right|rightof|simple_name|stable|succ|transaction|val|value)(?=[^\w-]|$)/i,null],["lit",/^\d+(?:_\d+)*(?:#[\w.\\]+#(?:[+-]?\d+(?:_\d+)*)?|(?:\.\d+(?:_\d+)*)?(?:e[+-]?\d+(?:_\d+)*)?)/i],
+["pln",/^(?:[a-z]\w*|\\[^\\]*\\)/i],["pun",/^[^\w\t\n\r "'\xa0][^\w\t\n\r "'\xa0-]*/]]),["vhdl","vhd"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-wiki.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-wiki.js
new file mode 100644
index 0000000..9b0b448
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-wiki.js
@@ -0,0 +1,2 @@
+PR.registerLangHandler(PR.createSimpleLexer([["pln",/^[\d\t a-gi-z\xa0]+/,null,"\t Â\xa0abcdefgijklmnopqrstuvwxyz0123456789"],["pun",/^[*=[\]^~]+/,null,"=*~^[]"]],[["lang-wiki.meta",/(?:^^|\r\n?|\n)(#[a-z]+)\b/],["lit",/^[A-Z][a-z][\da-z]+[A-Z][a-z][^\W_]+\b/],["lang-",/^{{{([\S\s]+?)}}}/],["lang-",/^`([^\n\r`]+)`/],["str",/^https?:\/\/[^\s#/?]*(?:\/[^\s#?]*)?(?:\?[^\s#]*)?(?:#\S*)?/i],["pln",/^(?:\r\n|[\S\s])[^\n\r#*=A-[^`h{~]*/]]),["wiki"]);
+PR.registerLangHandler(PR.createSimpleLexer([["kwd",/^#[a-z]+/i,null,"#"]],[]),["wiki.meta"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-xq.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-xq.js
new file mode 100644
index 0000000..e323ae3
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-xq.js
@@ -0,0 +1,3 @@
+PR.registerLangHandler(PR.createSimpleLexer([["var pln",/^\$[\w-]+/,null,"$"]],[["pln",/^[\s=][<>][\s=]/],["lit",/^@[\w-]+/],["tag",/^<\/?[a-z](?:[\w-.:]*\w)?|\/?>$/i],["com",/^\(:[\S\s]*?:\)/],["pln",/^[(),/;[\]{}]$/],["str",/^(?:"(?:[^"\\{]|\\[\S\s])*(?:"|$)|'(?:[^'\\{]|\\[\S\s])*(?:'|$))/,null,"\"'"],["kwd",/^(?:xquery|where|version|variable|union|typeswitch|treat|to|then|text|stable|sortby|some|self|schema|satisfies|returns|return|ref|processing-instruction|preceding-sibling|preceding|precedes|parent|only|of|node|namespace|module|let|item|intersect|instance|in|import|if|function|for|follows|following-sibling|following|external|except|every|else|element|descending|descendant-or-self|descendant|define|default|declare|comment|child|cast|case|before|attribute|assert|ascending|as|ancestor-or-self|ancestor|after|eq|order|by|or|and|schema-element|document-node|node|at)\b/],
+["typ",/^(?:xs:yearMonthDuration|xs:unsignedLong|xs:time|xs:string|xs:short|xs:QName|xs:Name|xs:long|xs:integer|xs:int|xs:gYearMonth|xs:gYear|xs:gMonthDay|xs:gDay|xs:float|xs:duration|xs:double|xs:decimal|xs:dayTimeDuration|xs:dateTime|xs:date|xs:byte|xs:boolean|xs:anyURI|xf:yearMonthDuration)\b/,null],["fun pln",/^(?:xp:dereference|xinc:node-expand|xinc:link-references|xinc:link-expand|xhtml:restructure|xhtml:clean|xhtml:add-lists|xdmp:zip-manifest|xdmp:zip-get|xdmp:zip-create|xdmp:xquery-version|xdmp:word-convert|xdmp:with-namespaces|xdmp:version|xdmp:value|xdmp:user-roles|xdmp:user-last-login|xdmp:user|xdmp:url-encode|xdmp:url-decode|xdmp:uri-is-file|xdmp:uri-format|xdmp:uri-content-type|xdmp:unquote|xdmp:unpath|xdmp:triggers-database|xdmp:trace|xdmp:to-json|xdmp:tidy|xdmp:subbinary|xdmp:strftime|xdmp:spawn-in|xdmp:spawn|xdmp:sleep|xdmp:shutdown|xdmp:set-session-field|xdmp:set-response-encoding|xdmp:set-response-content-type|xdmp:set-response-code|xdmp:set-request-time-limit|xdmp:set|xdmp:servers|xdmp:server-status|xdmp:server-name|xdmp:server|xdmp:security-database|xdmp:security-assert|xdmp:schema-database|xdmp:save|xdmp:role-roles|xdmp:role|xdmp:rethrow|xdmp:restart|xdmp:request-timestamp|xdmp:request-status|xdmp:request-cancel|xdmp:request|xdmp:redirect-response|xdmp:random|xdmp:quote|xdmp:query-trace|xdmp:query-meters|xdmp:product-edition|xdmp:privilege-roles|xdmp:privilege|xdmp:pretty-print|xdmp:powerpoint-convert|xdmp:platform|xdmp:permission|xdmp:pdf-convert|xdmp:path|xdmp:octal-to-integer|xdmp:node-uri|xdmp:node-replace|xdmp:node-kind|xdmp:node-insert-child|xdmp:node-insert-before|xdmp:node-insert-after|xdmp:node-delete|xdmp:node-database|xdmp:mul64|xdmp:modules-root|xdmp:modules-database|xdmp:merging|xdmp:merge-cancel|xdmp:merge|xdmp:md5|xdmp:logout|xdmp:login|xdmp:log-level|xdmp:log|xdmp:lock-release|xdmp:lock-acquire|xdmp:load|xdmp:invoke-in|xdmp:invoke|xdmp:integer-to-octal|xdmp:integer-to-hex|xdmp:http-put|xdmp:http-post|xdmp:http-options|xdmp:http-head|xdmp:http-get|xdmp:http-delete|xdmp:hosts|xdmp:host-status|xdmp:host-name|xdmp:host|xdmp:hex-to-integer|xdmp:hash64|xdmp:hash32|xdmp:has-privilege|xdmp:groups|xdmp:group-serves|xdmp:group-servers|xdmp:group-name|xdmp:group-hosts|xdmp:group|xdmp:get-session-field-names|xdmp:get-session-field|xdmp:get-response-encoding|xdmp:get-response-code|xdmp:get-request-username|xdmp:get-request-user|xdmp:get-request-url|xdmp:get-request-protocol|xdmp:get-request-path|xdmp:get-request-method|xdmp:get-request-header-names|xdmp:get-request-header|xdmp:get-request-field-names|xdmp:get-request-field-filename|xdmp:get-request-field-content-type|xdmp:get-request-field|xdmp:get-request-client-certificate|xdmp:get-request-client-address|xdmp:get-request-body|xdmp:get-current-user|xdmp:get-current-roles|xdmp:get|xdmp:function-name|xdmp:function-module|xdmp:function|xdmp:from-json|xdmp:forests|xdmp:forest-status|xdmp:forest-restore|xdmp:forest-restart|xdmp:forest-name|xdmp:forest-delete|xdmp:forest-databases|xdmp:forest-counts|xdmp:forest-clear|xdmp:forest-backup|xdmp:forest|xdmp:filesystem-file|xdmp:filesystem-directory|xdmp:exists|xdmp:excel-convert|xdmp:eval-in|xdmp:eval|xdmp:estimate|xdmp:email|xdmp:element-content-type|xdmp:elapsed-time|xdmp:document-set-quality|xdmp:document-set-property|xdmp:document-set-properties|xdmp:document-set-permissions|xdmp:document-set-collections|xdmp:document-remove-properties|xdmp:document-remove-permissions|xdmp:document-remove-collections|xdmp:document-properties|xdmp:document-locks|xdmp:document-load|xdmp:document-insert|xdmp:document-get-quality|xdmp:document-get-properties|xdmp:document-get-permissions|xdmp:document-get-collections|xdmp:document-get|xdmp:document-forest|xdmp:document-delete|xdmp:document-add-properties|xdmp:document-add-permissions|xdmp:document-add-collections|xdmp:directory-properties|xdmp:directory-locks|xdmp:directory-delete|xdmp:directory-create|xdmp:directory|xdmp:diacritic-less|xdmp:describe|xdmp:default-permissions|xdmp:default-collections|xdmp:databases|xdmp:database-restore-validate|xdmp:database-restore-status|xdmp:database-restore-cancel|xdmp:database-restore|xdmp:database-name|xdmp:database-forests|xdmp:database-backup-validate|xdmp:database-backup-status|xdmp:database-backup-purge|xdmp:database-backup-cancel|xdmp:database-backup|xdmp:database|xdmp:collection-properties|xdmp:collection-locks|xdmp:collection-delete|xdmp:collation-canonical-uri|xdmp:castable-as|xdmp:can-grant-roles|xdmp:base64-encode|xdmp:base64-decode|xdmp:architecture|xdmp:apply|xdmp:amp-roles|xdmp:amp|xdmp:add64|xdmp:add-response-header|xdmp:access|trgr:trigger-set-recursive|trgr:trigger-set-permissions|trgr:trigger-set-name|trgr:trigger-set-module|trgr:trigger-set-event|trgr:trigger-set-description|trgr:trigger-remove-permissions|trgr:trigger-module|trgr:trigger-get-permissions|trgr:trigger-enable|trgr:trigger-disable|trgr:trigger-database-online-event|trgr:trigger-data-event|trgr:trigger-add-permissions|trgr:remove-trigger|trgr:property-content|trgr:pre-commit|trgr:post-commit|trgr:get-trigger-by-id|trgr:get-trigger|trgr:document-scope|trgr:document-content|trgr:directory-scope|trgr:create-trigger|trgr:collection-scope|trgr:any-property-content|thsr:set-entry|thsr:remove-term|thsr:remove-synonym|thsr:remove-entry|thsr:query-lookup|thsr:lookup|thsr:load|thsr:insert|thsr:expand|thsr:add-synonym|spell:suggest-detailed|spell:suggest|spell:remove-word|spell:make-dictionary|spell:load|spell:levenshtein-distance|spell:is-correct|spell:insert|spell:double-metaphone|spell:add-word|sec:users-collection|sec:user-set-roles|sec:user-set-password|sec:user-set-name|sec:user-set-description|sec:user-set-default-permissions|sec:user-set-default-collections|sec:user-remove-roles|sec:user-privileges|sec:user-get-roles|sec:user-get-description|sec:user-get-default-permissions|sec:user-get-default-collections|sec:user-doc-permissions|sec:user-doc-collections|sec:user-add-roles|sec:unprotect-collection|sec:uid-for-name|sec:set-realm|sec:security-version|sec:security-namespace|sec:security-installed|sec:security-collection|sec:roles-collection|sec:role-set-roles|sec:role-set-name|sec:role-set-description|sec:role-set-default-permissions|sec:role-set-default-collections|sec:role-remove-roles|sec:role-privileges|sec:role-get-roles|sec:role-get-description|sec:role-get-default-permissions|sec:role-get-default-collections|sec:role-doc-permissions|sec:role-doc-collections|sec:role-add-roles|sec:remove-user|sec:remove-role-from-users|sec:remove-role-from-role|sec:remove-role-from-privileges|sec:remove-role-from-amps|sec:remove-role|sec:remove-privilege|sec:remove-amp|sec:protect-collection|sec:privileges-collection|sec:privilege-set-roles|sec:privilege-set-name|sec:privilege-remove-roles|sec:privilege-get-roles|sec:privilege-add-roles|sec:priv-doc-permissions|sec:priv-doc-collections|sec:get-user-names|sec:get-unique-elem-id|sec:get-role-names|sec:get-role-ids|sec:get-privilege|sec:get-distinct-permissions|sec:get-collection|sec:get-amp|sec:create-user-with-role|sec:create-user|sec:create-role|sec:create-privilege|sec:create-amp|sec:collections-collection|sec:collection-set-permissions|sec:collection-remove-permissions|sec:collection-get-permissions|sec:collection-add-permissions|sec:check-admin|sec:amps-collection|sec:amp-set-roles|sec:amp-remove-roles|sec:amp-get-roles|sec:amp-doc-permissions|sec:amp-doc-collections|sec:amp-add-roles|search:unparse|search:suggest|search:snippet|search:search|search:resolve-nodes|search:resolve|search:remove-constraint|search:parse|search:get-default-options|search:estimate|search:check-options|prof:value|prof:reset|prof:report|prof:invoke|prof:eval|prof:enable|prof:disable|prof:allowed|ppt:clean|pki:template-set-request|pki:template-set-name|pki:template-set-key-type|pki:template-set-key-options|pki:template-set-description|pki:template-in-use|pki:template-get-version|pki:template-get-request|pki:template-get-name|pki:template-get-key-type|pki:template-get-key-options|pki:template-get-id|pki:template-get-description|pki:need-certificate|pki:is-temporary|pki:insert-trusted-certificates|pki:insert-template|pki:insert-signed-certificates|pki:insert-certificate-revocation-list|pki:get-trusted-certificate-ids|pki:get-template-ids|pki:get-template-certificate-authority|pki:get-template-by-name|pki:get-template|pki:get-pending-certificate-requests-xml|pki:get-pending-certificate-requests-pem|pki:get-pending-certificate-request|pki:get-certificates-for-template-xml|pki:get-certificates-for-template|pki:get-certificates|pki:get-certificate-xml|pki:get-certificate-pem|pki:get-certificate|pki:generate-temporary-certificate-if-necessary|pki:generate-temporary-certificate|pki:generate-template-certificate-authority|pki:generate-certificate-request|pki:delete-template|pki:delete-certificate|pki:create-template|pdf:make-toc|pdf:insert-toc-headers|pdf:get-toc|pdf:clean|p:status-transition|p:state-transition|p:remove|p:pipelines|p:insert|p:get-by-id|p:get|p:execute|p:create|p:condition|p:collection|p:action|ooxml:runs-merge|ooxml:package-uris|ooxml:package-parts-insert|ooxml:package-parts|msword:clean|mcgm:polygon|mcgm:point|mcgm:geospatial-query-from-elements|mcgm:geospatial-query|mcgm:circle|math:tanh|math:tan|math:sqrt|math:sinh|math:sin|math:pow|math:modf|math:log10|math:log|math:ldexp|math:frexp|math:fmod|math:floor|math:fabs|math:exp|math:cosh|math:cos|math:ceil|math:atan2|math:atan|math:asin|math:acos|map:put|map:map|map:keys|map:get|map:delete|map:count|map:clear|lnk:to|lnk:remove|lnk:insert|lnk:get|lnk:from|lnk:create|kml:polygon|kml:point|kml:interior-polygon|kml:geospatial-query-from-elements|kml:geospatial-query|kml:circle|kml:box|gml:polygon|gml:point|gml:interior-polygon|gml:geospatial-query-from-elements|gml:geospatial-query|gml:circle|gml:box|georss:point|georss:geospatial-query|georss:circle|geo:polygon|geo:point|geo:interior-polygon|geo:geospatial-query-from-elements|geo:geospatial-query|geo:circle|geo:box|fn:zero-or-one|fn:years-from-duration|fn:year-from-dateTime|fn:year-from-date|fn:upper-case|fn:unordered|fn:true|fn:translate|fn:trace|fn:tokenize|fn:timezone-from-time|fn:timezone-from-dateTime|fn:timezone-from-date|fn:sum|fn:subtract-dateTimes-yielding-yearMonthDuration|fn:subtract-dateTimes-yielding-dayTimeDuration|fn:substring-before|fn:substring-after|fn:substring|fn:subsequence|fn:string-to-codepoints|fn:string-pad|fn:string-length|fn:string-join|fn:string|fn:static-base-uri|fn:starts-with|fn:seconds-from-time|fn:seconds-from-duration|fn:seconds-from-dateTime|fn:round-half-to-even|fn:round|fn:root|fn:reverse|fn:resolve-uri|fn:resolve-QName|fn:replace|fn:remove|fn:QName|fn:prefix-from-QName|fn:position|fn:one-or-more|fn:number|fn:not|fn:normalize-unicode|fn:normalize-space|fn:node-name|fn:node-kind|fn:nilled|fn:namespace-uri-from-QName|fn:namespace-uri-for-prefix|fn:namespace-uri|fn:name|fn:months-from-duration|fn:month-from-dateTime|fn:month-from-date|fn:minutes-from-time|fn:minutes-from-duration|fn:minutes-from-dateTime|fn:min|fn:max|fn:matches|fn:lower-case|fn:local-name-from-QName|fn:local-name|fn:last|fn:lang|fn:iri-to-uri|fn:insert-before|fn:index-of|fn:in-scope-prefixes|fn:implicit-timezone|fn:idref|fn:id|fn:hours-from-time|fn:hours-from-duration|fn:hours-from-dateTime|fn:floor|fn:false|fn:expanded-QName|fn:exists|fn:exactly-one|fn:escape-uri|fn:escape-html-uri|fn:error|fn:ends-with|fn:encode-for-uri|fn:empty|fn:document-uri|fn:doc-available|fn:doc|fn:distinct-values|fn:distinct-nodes|fn:default-collation|fn:deep-equal|fn:days-from-duration|fn:day-from-dateTime|fn:day-from-date|fn:data|fn:current-time|fn:current-dateTime|fn:current-date|fn:count|fn:contains|fn:concat|fn:compare|fn:collection|fn:codepoints-to-string|fn:codepoint-equal|fn:ceiling|fn:boolean|fn:base-uri|fn:avg|fn:adjust-time-to-timezone|fn:adjust-dateTime-to-timezone|fn:adjust-date-to-timezone|fn:abs|feed:unsubscribe|feed:subscription|feed:subscribe|feed:request|feed:item|feed:description|excel:clean|entity:enrich|dom:set-pipelines|dom:set-permissions|dom:set-name|dom:set-evaluation-context|dom:set-domain-scope|dom:set-description|dom:remove-pipeline|dom:remove-permissions|dom:remove|dom:get|dom:evaluation-context|dom:domains|dom:domain-scope|dom:create|dom:configuration-set-restart-user|dom:configuration-set-permissions|dom:configuration-set-evaluation-context|dom:configuration-set-default-domain|dom:configuration-get|dom:configuration-create|dom:collection|dom:add-pipeline|dom:add-permissions|dls:retention-rules|dls:retention-rule-remove|dls:retention-rule-insert|dls:retention-rule|dls:purge|dls:node-expand|dls:link-references|dls:link-expand|dls:documents-query|dls:document-versions-query|dls:document-version-uri|dls:document-version-query|dls:document-version-delete|dls:document-version-as-of|dls:document-version|dls:document-update|dls:document-unmanage|dls:document-set-quality|dls:document-set-property|dls:document-set-properties|dls:document-set-permissions|dls:document-set-collections|dls:document-retention-rules|dls:document-remove-properties|dls:document-remove-permissions|dls:document-remove-collections|dls:document-purge|dls:document-manage|dls:document-is-managed|dls:document-insert-and-manage|dls:document-include-query|dls:document-history|dls:document-get-permissions|dls:document-extract-part|dls:document-delete|dls:document-checkout-status|dls:document-checkout|dls:document-checkin|dls:document-add-properties|dls:document-add-permissions|dls:document-add-collections|dls:break-checkout|dls:author-query|dls:as-of-query|dbk:convert|dbg:wait|dbg:value|dbg:stopped|dbg:stop|dbg:step|dbg:status|dbg:stack|dbg:out|dbg:next|dbg:line|dbg:invoke|dbg:function|dbg:finish|dbg:expr|dbg:eval|dbg:disconnect|dbg:detach|dbg:continue|dbg:connect|dbg:clear|dbg:breakpoints|dbg:break|dbg:attached|dbg:attach|cvt:save-converted-documents|cvt:part-uri|cvt:destination-uri|cvt:basepath|cvt:basename|cts:words|cts:word-query-weight|cts:word-query-text|cts:word-query-options|cts:word-query|cts:word-match|cts:walk|cts:uris|cts:uri-match|cts:train|cts:tokenize|cts:thresholds|cts:stem|cts:similar-query-weight|cts:similar-query-nodes|cts:similar-query|cts:shortest-distance|cts:search|cts:score|cts:reverse-query-weight|cts:reverse-query-nodes|cts:reverse-query|cts:remainder|cts:registered-query-weight|cts:registered-query-options|cts:registered-query-ids|cts:registered-query|cts:register|cts:query|cts:quality|cts:properties-query-query|cts:properties-query|cts:polygon-vertices|cts:polygon|cts:point-longitude|cts:point-latitude|cts:point|cts:or-query-queries|cts:or-query|cts:not-query-weight|cts:not-query-query|cts:not-query|cts:near-query-weight|cts:near-query-queries|cts:near-query-options|cts:near-query-distance|cts:near-query|cts:highlight|cts:geospatial-co-occurrences|cts:frequency|cts:fitness|cts:field-words|cts:field-word-query-weight|cts:field-word-query-text|cts:field-word-query-options|cts:field-word-query-field-name|cts:field-word-query|cts:field-word-match|cts:entity-highlight|cts:element-words|cts:element-word-query-weight|cts:element-word-query-text|cts:element-word-query-options|cts:element-word-query-element-name|cts:element-word-query|cts:element-word-match|cts:element-values|cts:element-value-ranges|cts:element-value-query-weight|cts:element-value-query-text|cts:element-value-query-options|cts:element-value-query-element-name|cts:element-value-query|cts:element-value-match|cts:element-value-geospatial-co-occurrences|cts:element-value-co-occurrences|cts:element-range-query-weight|cts:element-range-query-value|cts:element-range-query-options|cts:element-range-query-operator|cts:element-range-query-element-name|cts:element-range-query|cts:element-query-query|cts:element-query-element-name|cts:element-query|cts:element-pair-geospatial-values|cts:element-pair-geospatial-value-match|cts:element-pair-geospatial-query-weight|cts:element-pair-geospatial-query-region|cts:element-pair-geospatial-query-options|cts:element-pair-geospatial-query-longitude-name|cts:element-pair-geospatial-query-latitude-name|cts:element-pair-geospatial-query-element-name|cts:element-pair-geospatial-query|cts:element-pair-geospatial-boxes|cts:element-geospatial-values|cts:element-geospatial-value-match|cts:element-geospatial-query-weight|cts:element-geospatial-query-region|cts:element-geospatial-query-options|cts:element-geospatial-query-element-name|cts:element-geospatial-query|cts:element-geospatial-boxes|cts:element-child-geospatial-values|cts:element-child-geospatial-value-match|cts:element-child-geospatial-query-weight|cts:element-child-geospatial-query-region|cts:element-child-geospatial-query-options|cts:element-child-geospatial-query-element-name|cts:element-child-geospatial-query-child-name|cts:element-child-geospatial-query|cts:element-child-geospatial-boxes|cts:element-attribute-words|cts:element-attribute-word-query-weight|cts:element-attribute-word-query-text|cts:element-attribute-word-query-options|cts:element-attribute-word-query-element-name|cts:element-attribute-word-query-attribute-name|cts:element-attribute-word-query|cts:element-attribute-word-match|cts:element-attribute-values|cts:element-attribute-value-ranges|cts:element-attribute-value-query-weight|cts:element-attribute-value-query-text|cts:element-attribute-value-query-options|cts:element-attribute-value-query-element-name|cts:element-attribute-value-query-attribute-name|cts:element-attribute-value-query|cts:element-attribute-value-match|cts:element-attribute-value-geospatial-co-occurrences|cts:element-attribute-value-co-occurrences|cts:element-attribute-range-query-weight|cts:element-attribute-range-query-value|cts:element-attribute-range-query-options|cts:element-attribute-range-query-operator|cts:element-attribute-range-query-element-name|cts:element-attribute-range-query-attribute-name|cts:element-attribute-range-query|cts:element-attribute-pair-geospatial-values|cts:element-attribute-pair-geospatial-value-match|cts:element-attribute-pair-geospatial-query-weight|cts:element-attribute-pair-geospatial-query-region|cts:element-attribute-pair-geospatial-query-options|cts:element-attribute-pair-geospatial-query-longitude-name|cts:element-attribute-pair-geospatial-query-latitude-name|cts:element-attribute-pair-geospatial-query-element-name|cts:element-attribute-pair-geospatial-query|cts:element-attribute-pair-geospatial-boxes|cts:document-query-uris|cts:document-query|cts:distance|cts:directory-query-uris|cts:directory-query-depth|cts:directory-query|cts:destination|cts:deregister|cts:contains|cts:confidence|cts:collections|cts:collection-query-uris|cts:collection-query|cts:collection-match|cts:classify|cts:circle-radius|cts:circle-center|cts:circle|cts:box-west|cts:box-south|cts:box-north|cts:box-east|cts:box|cts:bearing|cts:arc-intersection|cts:and-query-queries|cts:and-query-options|cts:and-query|cts:and-not-query-positive-query|cts:and-not-query-negative-query|cts:and-not-query|css:get|css:convert|cpf:success|cpf:failure|cpf:document-set-state|cpf:document-set-processing-status|cpf:document-set-last-updated|cpf:document-set-error|cpf:document-get-state|cpf:document-get-processing-status|cpf:document-get-last-updated|cpf:document-get-error|cpf:check-transition|alert:spawn-matching-actions|alert:rule-user-id-query|alert:rule-set-user-id|alert:rule-set-query|alert:rule-set-options|alert:rule-set-name|alert:rule-set-description|alert:rule-set-action|alert:rule-remove|alert:rule-name-query|alert:rule-insert|alert:rule-id-query|alert:rule-get-user-id|alert:rule-get-query|alert:rule-get-options|alert:rule-get-name|alert:rule-get-id|alert:rule-get-description|alert:rule-get-action|alert:rule-action-query|alert:remove-triggers|alert:make-rule|alert:make-log-action|alert:make-config|alert:make-action|alert:invoke-matching-actions|alert:get-my-rules|alert:get-all-rules|alert:get-actions|alert:find-matching-rules|alert:create-triggers|alert:config-set-uri|alert:config-set-trigger-ids|alert:config-set-options|alert:config-set-name|alert:config-set-description|alert:config-set-cpf-domain-names|alert:config-set-cpf-domain-ids|alert:config-insert|alert:config-get-uri|alert:config-get-trigger-ids|alert:config-get-options|alert:config-get-name|alert:config-get-id|alert:config-get-description|alert:config-get-cpf-domain-names|alert:config-get-cpf-domain-ids|alert:config-get|alert:config-delete|alert:action-set-options|alert:action-set-name|alert:action-set-module-root|alert:action-set-module-db|alert:action-set-module|alert:action-set-description|alert:action-remove|alert:action-insert|alert:action-get-options|alert:action-get-name|alert:action-get-module-root|alert:action-get-module-db|alert:action-get-module|alert:action-get-description|zero-or-one|years-from-duration|year-from-dateTime|year-from-date|upper-case|unordered|true|translate|trace|tokenize|timezone-from-time|timezone-from-dateTime|timezone-from-date|sum|subtract-dateTimes-yielding-yearMonthDuration|subtract-dateTimes-yielding-dayTimeDuration|substring-before|substring-after|substring|subsequence|string-to-codepoints|string-pad|string-length|string-join|string|static-base-uri|starts-with|seconds-from-time|seconds-from-duration|seconds-from-dateTime|round-half-to-even|round|root|reverse|resolve-uri|resolve-QName|replace|remove|QName|prefix-from-QName|position|one-or-more|number|not|normalize-unicode|normalize-space|node-name|node-kind|nilled|namespace-uri-from-QName|namespace-uri-for-prefix|namespace-uri|name|months-from-duration|month-from-dateTime|month-from-date|minutes-from-time|minutes-from-duration|minutes-from-dateTime|min|max|matches|lower-case|local-name-from-QName|local-name|last|lang|iri-to-uri|insert-before|index-of|in-scope-prefixes|implicit-timezone|idref|id|hours-from-time|hours-from-duration|hours-from-dateTime|floor|false|expanded-QName|exists|exactly-one|escape-uri|escape-html-uri|error|ends-with|encode-for-uri|empty|document-uri|doc-available|doc|distinct-values|distinct-nodes|default-collation|deep-equal|days-from-duration|day-from-dateTime|day-from-date|data|current-time|current-dateTime|current-date|count|contains|concat|compare|collection|codepoints-to-string|codepoint-equal|ceiling|boolean|base-uri|avg|adjust-time-to-timezone|adjust-dateTime-to-timezone|adjust-date-to-timezone|abs)\b/],
+["pln",/^[\w:-]+/],["pln",/^[\t\n\r \xa0]+/]]),["xq","xquery"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-yaml.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-yaml.js
new file mode 100644
index 0000000..c38729b
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/lang-yaml.js
@@ -0,0 +1,2 @@
+var a=null;
+PR.registerLangHandler(PR.createSimpleLexer([["pun",/^[:>?|]+/,a,":|>?"],["dec",/^%(?:YAML|TAG)[^\n\r#]+/,a,"%"],["typ",/^&\S+/,a,"&"],["typ",/^!\S*/,a,"!"],["str",/^"(?:[^"\\]|\\.)*(?:"|$)/,a,'"'],["str",/^'(?:[^']|'')*(?:'|$)/,a,"'"],["com",/^#[^\n\r]*/,a,"#"],["pln",/^\s+/,a," \t\r\n"]],[["dec",/^(?:---|\.\.\.)(?:[\n\r]|$)/],["pun",/^-/],["kwd",/^\w+:[\n\r ]/],["pln",/^\w+/]]),["yaml","yml"]);
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/prettify.css b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/prettify.css
new file mode 100644
index 0000000..d44b3a2
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/prettify.css
@@ -0,0 +1 @@
+.pln{color:#000}@media screen{.str{color:#080}.kwd{color:#008}.com{color:#800}.typ{color:#606}.lit{color:#066}.pun,.opn,.clo{color:#660}.tag{color:#008}.atn{color:#606}.atv{color:#080}.dec,.var{color:#606}.fun{color:red}}@media print,projection{.str{color:#060}.kwd{color:#006;font-weight:bold}.com{color:#600;font-style:italic}.typ{color:#404;font-weight:bold}.lit{color:#044}.pun,.opn,.clo{color:#440}.tag{color:#006;font-weight:bold}.atn{color:#404}.atv{color:#060}}pre.prettyprint{padding:2px;border:1px solid #888}ol.linenums{margin-top:0;margin-bottom:0}li.L0,li.L1,li.L2,li.L3,li.L5,li.L6,li.L7,li.L8{list-style-type:none}li.L1,li.L3,li.L5,li.L7,li.L9{background:#eee}
\ No newline at end of file
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/prettify.js b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/prettify.js
new file mode 100644
index 0000000..eef5ad7
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/static/prettify/prettify.js
@@ -0,0 +1,28 @@
+var q=null;window.PR_SHOULD_USE_CONTINUATION=!0;
+(function(){function L(a){function m(a){var f=a.charCodeAt(0);if(f!==92)return f;var b=a.charAt(1);return(f=r[b])?f:"0"<=b&&b<="7"?parseInt(a.substring(1),8):b==="u"||b==="x"?parseInt(a.substring(2),16):a.charCodeAt(1)}function e(a){if(a<32)return(a<16?"\\x0":"\\x")+a.toString(16);a=String.fromCharCode(a);if(a==="\\"||a==="-"||a==="["||a==="]")a="\\"+a;return a}function h(a){for(var f=a.substring(1,a.length-1).match(/\\u[\dA-Fa-f]{4}|\\x[\dA-Fa-f]{2}|\\[0-3][0-7]{0,2}|\\[0-7]{1,2}|\\[\S\s]|[^\\]/g),a=
+[],b=[],o=f[0]==="^",c=o?1:0,i=f.length;c<i;++c){var j=f[c];if(/\\[bdsw]/i.test(j))a.push(j);else{var j=m(j),d;c+2<i&&"-"===f[c+1]?(d=m(f[c+2]),c+=2):d=j;b.push([j,d]);d<65||j>122||(d<65||j>90||b.push([Math.max(65,j)|32,Math.min(d,90)|32]),d<97||j>122||b.push([Math.max(97,j)&-33,Math.min(d,122)&-33]))}}b.sort(function(a,f){return a[0]-f[0]||f[1]-a[1]});f=[];j=[NaN,NaN];for(c=0;c<b.length;++c)i=b[c],i[0]<=j[1]+1?j[1]=Math.max(j[1],i[1]):f.push(j=i);b=["["];o&&b.push("^");b.push.apply(b,a);for(c=0;c<
+f.length;++c)i=f[c],b.push(e(i[0])),i[1]>i[0]&&(i[1]+1>i[0]&&b.push("-"),b.push(e(i[1])));b.push("]");return b.join("")}function y(a){for(var f=a.source.match(/\[(?:[^\\\]]|\\[\S\s])*]|\\u[\dA-Fa-f]{4}|\\x[\dA-Fa-f]{2}|\\\d+|\\[^\dux]|\(\?[!:=]|[()^]|[^()[\\^]+/g),b=f.length,d=[],c=0,i=0;c<b;++c){var j=f[c];j==="("?++i:"\\"===j.charAt(0)&&(j=+j.substring(1))&&j<=i&&(d[j]=-1)}for(c=1;c<d.length;++c)-1===d[c]&&(d[c]=++t);for(i=c=0;c<b;++c)j=f[c],j==="("?(++i,d[i]===void 0&&(f[c]="(?:")):"\\"===j.charAt(0)&&
+(j=+j.substring(1))&&j<=i&&(f[c]="\\"+d[i]);for(i=c=0;c<b;++c)"^"===f[c]&&"^"!==f[c+1]&&(f[c]="");if(a.ignoreCase&&s)for(c=0;c<b;++c)j=f[c],a=j.charAt(0),j.length>=2&&a==="["?f[c]=h(j):a!=="\\"&&(f[c]=j.replace(/[A-Za-z]/g,function(a){a=a.charCodeAt(0);return"["+String.fromCharCode(a&-33,a|32)+"]"}));return f.join("")}for(var t=0,s=!1,l=!1,p=0,d=a.length;p<d;++p){var g=a[p];if(g.ignoreCase)l=!0;else if(/[a-z]/i.test(g.source.replace(/\\u[\da-f]{4}|\\x[\da-f]{2}|\\[^UXux]/gi,""))){s=!0;l=!1;break}}for(var r=
+{b:8,t:9,n:10,v:11,f:12,r:13},n=[],p=0,d=a.length;p<d;++p){g=a[p];if(g.global||g.multiline)throw Error(""+g);n.push("(?:"+y(g)+")")}return RegExp(n.join("|"),l?"gi":"g")}function M(a){function m(a){switch(a.nodeType){case 1:if(e.test(a.className))break;for(var g=a.firstChild;g;g=g.nextSibling)m(g);g=a.nodeName;if("BR"===g||"LI"===g)h[s]="\n",t[s<<1]=y++,t[s++<<1|1]=a;break;case 3:case 4:g=a.nodeValue,g.length&&(g=p?g.replace(/\r\n?/g,"\n"):g.replace(/[\t\n\r ]+/g," "),h[s]=g,t[s<<1]=y,y+=g.length,
+t[s++<<1|1]=a)}}var e=/(?:^|\s)nocode(?:\s|$)/,h=[],y=0,t=[],s=0,l;a.currentStyle?l=a.currentStyle.whiteSpace:window.getComputedStyle&&(l=document.defaultView.getComputedStyle(a,q).getPropertyValue("white-space"));var p=l&&"pre"===l.substring(0,3);m(a);return{a:h.join("").replace(/\n$/,""),c:t}}function B(a,m,e,h){m&&(a={a:m,d:a},e(a),h.push.apply(h,a.e))}function x(a,m){function e(a){for(var l=a.d,p=[l,"pln"],d=0,g=a.a.match(y)||[],r={},n=0,z=g.length;n<z;++n){var f=g[n],b=r[f],o=void 0,c;if(typeof b===
+"string")c=!1;else{var i=h[f.charAt(0)];if(i)o=f.match(i[1]),b=i[0];else{for(c=0;c<t;++c)if(i=m[c],o=f.match(i[1])){b=i[0];break}o||(b="pln")}if((c=b.length>=5&&"lang-"===b.substring(0,5))&&!(o&&typeof o[1]==="string"))c=!1,b="src";c||(r[f]=b)}i=d;d+=f.length;if(c){c=o[1];var j=f.indexOf(c),k=j+c.length;o[2]&&(k=f.length-o[2].length,j=k-c.length);b=b.substring(5);B(l+i,f.substring(0,j),e,p);B(l+i+j,c,C(b,c),p);B(l+i+k,f.substring(k),e,p)}else p.push(l+i,b)}a.e=p}var h={},y;(function(){for(var e=a.concat(m),
+l=[],p={},d=0,g=e.length;d<g;++d){var r=e[d],n=r[3];if(n)for(var k=n.length;--k>=0;)h[n.charAt(k)]=r;r=r[1];n=""+r;p.hasOwnProperty(n)||(l.push(r),p[n]=q)}l.push(/[\S\s]/);y=L(l)})();var t=m.length;return e}function u(a){var m=[],e=[];a.tripleQuotedStrings?m.push(["str",/^(?:'''(?:[^'\\]|\\[\S\s]|''?(?=[^']))*(?:'''|$)|"""(?:[^"\\]|\\[\S\s]|""?(?=[^"]))*(?:"""|$)|'(?:[^'\\]|\\[\S\s])*(?:'|$)|"(?:[^"\\]|\\[\S\s])*(?:"|$))/,q,"'\""]):a.multiLineStrings?m.push(["str",/^(?:'(?:[^'\\]|\\[\S\s])*(?:'|$)|"(?:[^"\\]|\\[\S\s])*(?:"|$)|`(?:[^\\`]|\\[\S\s])*(?:`|$))/,
+q,"'\"`"]):m.push(["str",/^(?:'(?:[^\n\r'\\]|\\.)*(?:'|$)|"(?:[^\n\r"\\]|\\.)*(?:"|$))/,q,"\"'"]);a.verbatimStrings&&e.push(["str",/^@"(?:[^"]|"")*(?:"|$)/,q]);var h=a.hashComments;h&&(a.cStyleComments?(h>1?m.push(["com",/^#(?:##(?:[^#]|#(?!##))*(?:###|$)|.*)/,q,"#"]):m.push(["com",/^#(?:(?:define|elif|else|endif|error|ifdef|include|ifndef|line|pragma|undef|warning)\b|[^\n\r]*)/,q,"#"]),e.push(["str",/^<(?:(?:(?:\.\.\/)*|\/?)(?:[\w-]+(?:\/[\w-]+)+)?[\w-]+\.h|[a-z]\w*)>/,q])):m.push(["com",/^#[^\n\r]*/,
+q,"#"]));a.cStyleComments&&(e.push(["com",/^\/\/[^\n\r]*/,q]),e.push(["com",/^\/\*[\S\s]*?(?:\*\/|$)/,q]));a.regexLiterals&&e.push(["lang-regex",/^(?:^^\.?|[!+-]|!=|!==|#|%|%=|&|&&|&&=|&=|\(|\*|\*=|\+=|,|-=|->|\/|\/=|:|::|;|<|<<|<<=|<=|=|==|===|>|>=|>>|>>=|>>>|>>>=|[?@[^]|\^=|\^\^|\^\^=|{|\||\|=|\|\||\|\|=|~|break|case|continue|delete|do|else|finally|instanceof|return|throw|try|typeof)\s*(\/(?=[^*/])(?:[^/[\\]|\\[\S\s]|\[(?:[^\\\]]|\\[\S\s])*(?:]|$))+\/)/]);(h=a.types)&&e.push(["typ",h]);a=(""+a.keywords).replace(/^ | $/g,
+"");a.length&&e.push(["kwd",RegExp("^(?:"+a.replace(/[\s,]+/g,"|")+")\\b"),q]);m.push(["pln",/^\s+/,q," \r\n\t\xa0"]);e.push(["lit",/^@[$_a-z][\w$@]*/i,q],["typ",/^(?:[@_]?[A-Z]+[a-z][\w$@]*|\w+_t\b)/,q],["pln",/^[$_a-z][\w$@]*/i,q],["lit",/^(?:0x[\da-f]+|(?:\d(?:_\d+)*\d*(?:\.\d*)?|\.\d\+)(?:e[+-]?\d+)?)[a-z]*/i,q,"0123456789"],["pln",/^\\[\S\s]?/,q],["pun",/^.[^\s\w"-$'./@\\`]*/,q]);return x(m,e)}function D(a,m){function e(a){switch(a.nodeType){case 1:if(k.test(a.className))break;if("BR"===a.nodeName)h(a),
+a.parentNode&&a.parentNode.removeChild(a);else for(a=a.firstChild;a;a=a.nextSibling)e(a);break;case 3:case 4:if(p){var b=a.nodeValue,d=b.match(t);if(d){var c=b.substring(0,d.index);a.nodeValue=c;(b=b.substring(d.index+d[0].length))&&a.parentNode.insertBefore(s.createTextNode(b),a.nextSibling);h(a);c||a.parentNode.removeChild(a)}}}}function h(a){function b(a,d){var e=d?a.cloneNode(!1):a,f=a.parentNode;if(f){var f=b(f,1),g=a.nextSibling;f.appendChild(e);for(var h=g;h;h=g)g=h.nextSibling,f.appendChild(h)}return e}
+for(;!a.nextSibling;)if(a=a.parentNode,!a)return;for(var a=b(a.nextSibling,0),e;(e=a.parentNode)&&e.nodeType===1;)a=e;d.push(a)}var k=/(?:^|\s)nocode(?:\s|$)/,t=/\r\n?|\n/,s=a.ownerDocument,l;a.currentStyle?l=a.currentStyle.whiteSpace:window.getComputedStyle&&(l=s.defaultView.getComputedStyle(a,q).getPropertyValue("white-space"));var p=l&&"pre"===l.substring(0,3);for(l=s.createElement("LI");a.firstChild;)l.appendChild(a.firstChild);for(var d=[l],g=0;g<d.length;++g)e(d[g]);m===(m|0)&&d[0].setAttribute("value",
+m);var r=s.createElement("OL");r.className="linenums";for(var n=Math.max(0,m-1|0)||0,g=0,z=d.length;g<z;++g)l=d[g],l.className="L"+(g+n)%10,l.firstChild||l.appendChild(s.createTextNode("\xa0")),r.appendChild(l);a.appendChild(r)}function k(a,m){for(var e=m.length;--e>=0;){var h=m[e];A.hasOwnProperty(h)?window.console&&console.warn("cannot override language handler %s",h):A[h]=a}}function C(a,m){if(!a||!A.hasOwnProperty(a))a=/^\s*</.test(m)?"default-markup":"default-code";return A[a]}function E(a){var m=
+a.g;try{var e=M(a.h),h=e.a;a.a=h;a.c=e.c;a.d=0;C(m,h)(a);var k=/\bMSIE\b/.test(navigator.userAgent),m=/\n/g,t=a.a,s=t.length,e=0,l=a.c,p=l.length,h=0,d=a.e,g=d.length,a=0;d[g]=s;var r,n;for(n=r=0;n<g;)d[n]!==d[n+2]?(d[r++]=d[n++],d[r++]=d[n++]):n+=2;g=r;for(n=r=0;n<g;){for(var z=d[n],f=d[n+1],b=n+2;b+2<=g&&d[b+1]===f;)b+=2;d[r++]=z;d[r++]=f;n=b}for(d.length=r;h<p;){var o=l[h+2]||s,c=d[a+2]||s,b=Math.min(o,c),i=l[h+1],j;if(i.nodeType!==1&&(j=t.substring(e,b))){k&&(j=j.replace(m,"\r"));i.nodeValue=
+j;var u=i.ownerDocument,v=u.createElement("SPAN");v.className=d[a+1];var x=i.parentNode;x.replaceChild(v,i);v.appendChild(i);e<o&&(l[h+1]=i=u.createTextNode(t.substring(b,o)),x.insertBefore(i,v.nextSibling))}e=b;e>=o&&(h+=2);e>=c&&(a+=2)}}catch(w){"console"in window&&console.log(w&&w.stack?w.stack:w)}}var v=["break,continue,do,else,for,if,return,while"],w=[[v,"auto,case,char,const,default,double,enum,extern,float,goto,int,long,register,short,signed,sizeof,static,struct,switch,typedef,union,unsigned,void,volatile"],
+"catch,class,delete,false,import,new,operator,private,protected,public,this,throw,true,try,typeof"],F=[w,"alignof,align_union,asm,axiom,bool,concept,concept_map,const_cast,constexpr,decltype,dynamic_cast,explicit,export,friend,inline,late_check,mutable,namespace,nullptr,reinterpret_cast,static_assert,static_cast,template,typeid,typename,using,virtual,where"],G=[w,"abstract,boolean,byte,extends,final,finally,implements,import,instanceof,null,native,package,strictfp,super,synchronized,throws,transient"],
+H=[G,"as,base,by,checked,decimal,delegate,descending,dynamic,event,fixed,foreach,from,group,implicit,in,interface,internal,into,is,lock,object,out,override,orderby,params,partial,readonly,ref,sbyte,sealed,stackalloc,string,select,uint,ulong,unchecked,unsafe,ushort,var"],w=[w,"debugger,eval,export,function,get,null,set,undefined,var,with,Infinity,NaN"],I=[v,"and,as,assert,class,def,del,elif,except,exec,finally,from,global,import,in,is,lambda,nonlocal,not,or,pass,print,raise,try,with,yield,False,True,None"],
+J=[v,"alias,and,begin,case,class,def,defined,elsif,end,ensure,false,in,module,next,nil,not,or,redo,rescue,retry,self,super,then,true,undef,unless,until,when,yield,BEGIN,END"],v=[v,"case,done,elif,esac,eval,fi,function,in,local,set,then,until"],K=/^(DIR|FILE|vector|(de|priority_)?queue|list|stack|(const_)?iterator|(multi)?(set|map)|bitset|u?(int|float)\d*)/,N=/\S/,O=u({keywords:[F,H,w,"caller,delete,die,do,dump,elsif,eval,exit,foreach,for,goto,if,import,last,local,my,next,no,our,print,package,redo,require,sub,undef,unless,until,use,wantarray,while,BEGIN,END"+
+I,J,v],hashComments:!0,cStyleComments:!0,multiLineStrings:!0,regexLiterals:!0}),A={};k(O,["default-code"]);k(x([],[["pln",/^[^<?]+/],["dec",/^<!\w[^>]*(?:>|$)/],["com",/^<\!--[\S\s]*?(?:--\>|$)/],["lang-",/^<\?([\S\s]+?)(?:\?>|$)/],["lang-",/^<%([\S\s]+?)(?:%>|$)/],["pun",/^(?:<[%?]|[%?]>)/],["lang-",/^<xmp\b[^>]*>([\S\s]+?)<\/xmp\b[^>]*>/i],["lang-js",/^<script\b[^>]*>([\S\s]*?)(<\/script\b[^>]*>)/i],["lang-css",/^<style\b[^>]*>([\S\s]*?)(<\/style\b[^>]*>)/i],["lang-in.tag",/^(<\/?[a-z][^<>]*>)/i]]),
+["default-markup","htm","html","mxml","xhtml","xml","xsl"]);k(x([["pln",/^\s+/,q," \t\r\n"],["atv",/^(?:"[^"]*"?|'[^']*'?)/,q,"\"'"]],[["tag",/^^<\/?[a-z](?:[\w-.:]*\w)?|\/?>$/i],["atn",/^(?!style[\s=]|on)[a-z](?:[\w:-]*\w)?/i],["lang-uq.val",/^=\s*([^\s"'>]*(?:[^\s"'/>]|\/(?=\s)))/],["pun",/^[/<->]+/],["lang-js",/^on\w+\s*=\s*"([^"]+)"/i],["lang-js",/^on\w+\s*=\s*'([^']+)'/i],["lang-js",/^on\w+\s*=\s*([^\s"'>]+)/i],["lang-css",/^style\s*=\s*"([^"]+)"/i],["lang-css",/^style\s*=\s*'([^']+)'/i],["lang-css",
+/^style\s*=\s*([^\s"'>]+)/i]]),["in.tag"]);k(x([],[["atv",/^[\S\s]+/]]),["uq.val"]);k(u({keywords:F,hashComments:!0,cStyleComments:!0,types:K}),["c","cc","cpp","cxx","cyc","m"]);k(u({keywords:"null,true,false"}),["json"]);k(u({keywords:H,hashComments:!0,cStyleComments:!0,verbatimStrings:!0,types:K}),["cs"]);k(u({keywords:G,cStyleComments:!0}),["java"]);k(u({keywords:v,hashComments:!0,multiLineStrings:!0}),["bsh","csh","sh"]);k(u({keywords:I,hashComments:!0,multiLineStrings:!0,tripleQuotedStrings:!0}),
+["cv","py"]);k(u({keywords:"caller,delete,die,do,dump,elsif,eval,exit,foreach,for,goto,if,import,last,local,my,next,no,our,print,package,redo,require,sub,undef,unless,until,use,wantarray,while,BEGIN,END",hashComments:!0,multiLineStrings:!0,regexLiterals:!0}),["perl","pl","pm"]);k(u({keywords:J,hashComments:!0,multiLineStrings:!0,regexLiterals:!0}),["rb"]);k(u({keywords:w,cStyleComments:!0,regexLiterals:!0}),["js"]);k(u({keywords:"all,and,by,catch,class,else,extends,false,finally,for,if,in,is,isnt,loop,new,no,not,null,of,off,on,or,return,super,then,true,try,unless,until,when,while,yes",
+hashComments:3,cStyleComments:!0,multilineStrings:!0,tripleQuotedStrings:!0,regexLiterals:!0}),["coffee"]);k(x([],[["str",/^[\S\s]+/]]),["regex"]);window.prettyPrintOne=function(a,m,e){var h=document.createElement("PRE");h.innerHTML=a;e&&D(h,e);E({g:m,i:e,h:h});return h.innerHTML};window.prettyPrint=function(a){function m(){for(var e=window.PR_SHOULD_USE_CONTINUATION?l.now()+250:Infinity;p<h.length&&l.now()<e;p++){var n=h[p],k=n.className;if(k.indexOf("prettyprint")>=0){var k=k.match(g),f,b;if(b=
+!k){b=n;for(var o=void 0,c=b.firstChild;c;c=c.nextSibling)var i=c.nodeType,o=i===1?o?b:c:i===3?N.test(c.nodeValue)?b:o:o;b=(f=o===b?void 0:o)&&"CODE"===f.tagName}b&&(k=f.className.match(g));k&&(k=k[1]);b=!1;for(o=n.parentNode;o;o=o.parentNode)if((o.tagName==="pre"||o.tagName==="code"||o.tagName==="xmp")&&o.className&&o.className.indexOf("prettyprint")>=0){b=!0;break}b||((b=(b=n.className.match(/\blinenums\b(?::(\d+))?/))?b[1]&&b[1].length?+b[1]:!0:!1)&&D(n,b),d={g:k,h:n,i:b},E(d))}}p<h.length?setTimeout(m,
+250):a&&a()}for(var e=[document.getElementsByTagName("pre"),document.getElementsByTagName("code"),document.getElementsByTagName("xmp")],h=[],k=0;k<e.length;++k)for(var t=0,s=e[k].length;t<s;++t)h.push(e[k][t]);var e=q,l=Date;l.now||(l={now:function(){return+new Date}});var p=0,d,g=/\blang(?:uage)?-([\w.]+)(?!\S)/;m()};window.PR={createSimpleLexer:x,registerLangHandler:k,sourceDecorator:u,PR_ATTRIB_NAME:"atn",PR_ATTRIB_VALUE:"atv",PR_COMMENT:"com",PR_DECLARATION:"dec",PR_KEYWORD:"kwd",PR_LITERAL:"lit",
+PR_NOCODE:"nocode",PR_PLAIN:"pln",PR_PUNCTUATION:"pun",PR_SOURCE:"src",PR_STRING:"str",PR_TAG:"tag",PR_TYPE:"typ"}})();
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/Common.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/Common.soy
new file mode 100644
index 0000000..4297d31
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/Common.soy
@@ -0,0 +1,95 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * Common header for Gitiles.
+ *
+ * @param title title for this page. Always suffixed with repository name and a
+ *     sitewide title.
+ * @param? repositoryName repository name for this page, if applicable.
+ * @param? menuEntries optional list of menu entries with "text" and optional
+ *     "url" keys.
+ * @param breadcrumbs navigation breadcrumbs for this page.
+ * @param? css optional list of CSS URLs to include.
+ * @param? js optional list of Javascript URLs to include.
+ * @param? onLoad optional Javascript to execute in the body's onLoad handler.
+ *     Warning: not autoescaped.
+ */
+{template .header}
+<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">
+<html>
+<head>
+  <title>
+    {$title}
+    {if $repositoryName}
+      {sp}- {$repositoryName}
+    {/if}
+    {sp}- {msg desc="name of the application"}Git at Google{/msg}
+  </title>
+  <link rel="stylesheet" type="text/css" href="//www.google.com/css/go.css" />
+
+  {if $css and length($css)}
+    {foreach $url in $css}
+      <link rel="stylesheet" type="text/css" href="{$url}" />
+    {/foreach}
+  {/if}
+  // Include default CSS after custom CSS so it can override defaults in third-
+  // party stylesheets (e.g. prettify).
+  <link rel="stylesheet" type="text/css" href="{gitiles.CSS_URL}" />
+
+  {if $js and length($js)}
+    {foreach $url in $js}
+      <script src="{$url}" type="text/javascript"></script>
+    {/foreach}
+  {/if}
+</head>
+<body {if $onLoad}onload="{$onLoad|id}"{/if}>
+  {call .customHeader /}
+
+  {if $menuEntries and length($menuEntries)}
+    <div class="menu">
+    {foreach $entry in $menuEntries}
+      {sp}
+      {if $entry.url}
+        <a href="{$entry.url}"{if not isLast($entry)} class="entry"{/if}>{$entry.text}</a>
+      {else}
+        <span{if not isLast($entry)} class="entry"{/if}>{$entry.text}</span>
+      {/if}
+    {/foreach}
+    {sp}
+    </div>
+  {/if}
+
+  {if $breadcrumbs and length($breadcrumbs)}
+    <div class="breadcrumbs">
+      {foreach $entry in $breadcrumbs}
+        {if not isFirst($entry)}{sp}/{sp}{/if}
+        {if not isLast($entry)}
+          <a href="{$entry.url}">{$entry.text}</a>
+        {else}
+          {$entry.text}
+        {/if}
+      {/foreach}
+    </div>
+  {/if}
+{/template}
+
+/**
+ * Standard footer.
+ */
+{template .footer}
+</body>
+</html>
+{/template}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/DefaultCustomTemplates.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/DefaultCustomTemplates.soy
new file mode 100644
index 0000000..a1f5d9e
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/DefaultCustomTemplates.soy
@@ -0,0 +1,21 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * Default custom header implementation for Gitiles.
+ */
+{template .customHeader}
+<h1>{msg desc="short name of the application"}Gitiles{/msg}</h1>
+{/template}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/DiffDetail.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/DiffDetail.soy
new file mode 100644
index 0000000..dfd8af2
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/DiffDetail.soy
@@ -0,0 +1,49 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * Detail page showing diffs for a single commit.
+ *
+ * @param title human-readable revision name.
+ * @param repositoryName name of this repository.
+ * @param? menuEntries menu entries.
+ * @param breadcrumbs breadcrumbs for this page.
+ * @param? commit optional commit for which diffs are displayed, with keys
+ *     corresponding to the gitiles.commitDetail template (minus "diffTree").
+ */
+{template .diffDetail}
+{call .header data="all" /}
+
+{if $commit}
+  {call .commitDetail data="$commit" /}
+{/if}
+<div id="DIFF_OUTPUT_BLOCK" />
+
+{call .footer /}
+{/template}
+
+/**
+ * File header for a single unified diff patch.
+ *
+ * @param first the first line of the header, with no trailing LF.
+ * @param rest remaining lines of the header, if any.
+ * @param fileIndex position of the file within the difference.
+ */
+{template .diffHeader}
+<pre class="diff-header">
+<a name="F{$fileIndex}" class="diff-git">{$first}</a>{\n}
+{$rest}
+</pre>
+{/template}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/HostIndex.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/HostIndex.soy
new file mode 100644
index 0000000..1ee9616
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/HostIndex.soy
@@ -0,0 +1,77 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * HTML page for /.
+ *
+ * @param hostName host name.
+ * @param? menuEntries menu entries.
+ * @param baseUrl base URL for repositories.
+ * @param repositories list of repository description maps with name, cloneUrl,
+ *     and optional description values.
+ */
+{template .hostIndex}
+{call .header}
+  {param title: $hostName ? $hostName + ' Git repositories' : 'Git repositories' /}
+  {param menuEntries: $menuEntries /}
+  {param breadcrumbs: null /}
+{/call}
+
+{if length($repositories)}
+
+  <h2>
+    {msg desc="Git repositories available on the host"}
+      {$hostName} Git repositories
+    {/msg}
+  </h2>
+
+  <div class="instructions">
+    {msg desc="description on how to use this repository"}
+    To clone one of these repositories, install{sp}
+    <a href="http://www.git-scm.com/">git</a>, and run:
+    <pre>git clone {$baseUrl}<em>name</em></pre>
+    {/msg}
+  </div>
+
+  <table class="list">
+    <tr class="no-hover">
+      <th width="25%">
+        {msg desc="column header for repository name"}
+          Name
+        {/msg}
+      </th>
+      <th>
+        {msg desc="column header for repository description"}
+          Description
+        {/msg}
+      </th>
+    </tr>
+    {foreach $repo in $repositories}
+      <tr>
+        <td>
+          <a href="{$repo.url}">{$repo.name}</a>
+        </td>
+        <td>{$repo.description}</td>
+      </tr>
+    {/foreach}
+  </table>
+  <div class="footer">
+    <a href="?format=TEXT">{msg desc="text format"}TXT{/msg}</a>
+    {sp}
+    <a href="?format=JSON">{msg desc="JSON format"}JSON{/msg}</a>
+  </div>
+{/if}
+{call .footer /}
+{/template}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/LogDetail.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/LogDetail.soy
new file mode 100644
index 0000000..b9ae9c7
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/LogDetail.soy
@@ -0,0 +1,95 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * Detail page showing a shortlog for a commit.
+ *
+ * @param title human-readable revision name.
+ * @param repositoryName name of this repository.
+ * @param? menuEntries menu entries.
+ * @param breadcrumbs breadcrumbs for this page.
+ * @param? tags optional list of tags encountered when peeling this object, with
+ *     keys corresponding to gitiles.tagDetail.
+ * @param entries list of log entries; see .logEntry.
+ * @param? nextUrl URL for the next page of results.
+ * @param? previousUrl URL for the previous page of results.
+ */
+{template .logDetail}
+{call .header data="all" /}
+
+{if $tags}
+  {foreach $tag in $tags}
+    {call gitiles.tagDetail data="$tag" /}
+  {/foreach}
+{/if}
+
+{if $previousUrl}
+  <div class="log-nav">
+    <a href="{$previousUrl}">{msg desc="text for previous URL"}&laquo; Previous{/msg}</a>
+  </div>
+{/if}
+
+{if length($entries)}
+  <ol class="shortlog">
+    {foreach $entry in $entries}
+      <li{if $previousUrl and isFirst($entry)} class="first"{/if}>
+        {call .logEntry data="$entry" /}
+      </li>
+    {/foreach}
+  </ol>
+{else}
+  <p>{msg desc="informational text for when the log is empty"}No commits.{/msg}</p>
+{/if}
+
+{if $nextUrl}
+  <div class="log-nav">
+    <a href="{$nextUrl}">{msg desc="text for next URL"}Next &raquo;{/msg}</a>
+  </div>
+{/if}
+
+{call .footer /}
+{/template}
+
+/**
+ * Single shortlog entry.
+ *
+ * @param abbrevSha abbreviated SHA-1.
+ * @param url URL to commit detail page.
+ * @param shortMessage short commit message.
+ * @param author author information with at least "name" and "relativeTime" keys.
+ * @param branches list of branches for this entry, with "name" and "url" keys.
+ * @param tags list of tags for this entry, with "name" and "url" keys.
+ */
+{template .logEntry}
+<a href="{$url}">
+  <span class="sha1">{$abbrevSha}</span>
+  // nbsp instad of CSS padding/margin because those cause a break in the
+  // underline.
+  &nbsp;
+  {sp}<span class="commit-message">{$shortMessage}</span>
+</a>
+{sp}<span class="author">{msg desc="commit author name"}by {$author.name}{/msg}</span>
+{sp}<span class="time">- {$author.relativeTime}</span>
+{if length($branches)}
+  {foreach $branch in $branches}
+    {sp}<a href="{$branch.url}" class="branch-label">{$branch.name}</a>
+  {/foreach}
+{/if}
+{if length($tags)}
+  {foreach $tag in $tags}
+    {sp}<a href="{$tag.url}" class="tag-label">{$tag.name}</a>
+  {/foreach}
+{/if}
+{/template}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/ObjectDetail.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/ObjectDetail.soy
new file mode 100644
index 0000000..badf43c
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/ObjectDetail.soy
@@ -0,0 +1,294 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * Detailed listing of a commit.
+ *
+ * @param author map with "name", "email", and "time" keys for the commit author.
+ * @param committer map with "name", "email", and "time" keys for the committer.
+ * @param sha commit SHA-1.
+ * @param tree tree SHA-1.
+ * @param treeUrl tree URL.
+ * @param parents list of parent objects with the following keys:
+ *     sha: SHA-1.
+ *     url: URL to view the parent commit.
+ *     diffUrl: URL to display diffs relative to this parent.
+ * @param message list of commit message parts, where each part contains:
+ *     text: raw text of the part.
+ *     url: optional URL that should be linked to from the part.
+ * @param diffTree list of changed tree entries with the following keys:
+ *     changeType: string matching an org.eclipse.jgit.diff.DiffEntry.ChangeType
+ *         constant.
+ *     path: (new) path of the tree entry.
+ *     oldPath: old path, only for renames and copies.
+ *     url: URL to a detail page for the tree entry.
+ *     diffUrl: URL to a diff page for the tree entry's diff in this commit.
+ * @param logUrl URL to a log page starting at this commit.
+ */
+{template .commitDetail}
+<div class="git-commit">
+  <table>
+    <tr>
+      <th>{msg desc="Header for commit SHA entry"}commit{/msg}</th>
+      <td class="sha">
+        {$sha}
+        <span class="log-link">
+          [<a href="{$logUrl}">{msg desc="text for the log link"}log{/msg}</a>]
+        </span>
+      </td>
+      <td>{sp}</td>
+    </tr>
+    <tr>
+      <th>{msg desc="Header for commit author"}author{/msg}</th>
+      <td>{call .person_ data="$author" /}</td>
+      <td>{$author.time}</td>
+    </tr>
+    <tr>
+      <th>{msg desc="Header for committer"}committer{/msg}</th>
+      <td>{call .person_ data="$committer" /}</td>
+      <td>{$committer.time}</td>
+    </tr>
+    <tr>
+      <th>{msg desc="Header for tree SHA entry"}tree{/msg}</th>
+      <td class="sha"><a href="{$treeUrl}">{$tree}</a></td>
+    </tr>
+    {foreach $parent in $parents}
+      <tr>
+        <th>{msg desc="Header for parent SHA"}parent{/msg}</th>
+        <td>
+          <a href="{$parent.url}">{$parent.sha}</a>
+          <span class="diff-link">
+            [<a href="{$parent.diffUrl}">{msg desc="text for the parent diff link"}diff{/msg}</a>]
+          </span>
+        </td>
+      </tr>
+    {/foreach}
+  </table>
+</div>
+{call .message_}
+  {param className: 'commit-message' /}
+  {param message: $message /}
+{/call}
+
+{if $diffTree and length($diffTree)}
+  <ul class="diff-tree">
+    {foreach $entry in $diffTree}
+      <li>
+        <a href="{$entry.url}">{$entry.path}</a>
+        {switch $entry.changeType}
+          {case 'ADD'}
+            <span class="add">
+              {msg desc="Text for a new tree entry"}
+                [Added - <a href="{$entry.diffUrl}">diff</a>]
+              {/msg}
+            </span>
+          {case 'MODIFY'}
+            <span class="modify">
+              {msg desc="Text for a modified tree entry"}
+                [<a href="{$entry.diffUrl}">diff</a>]
+              {/msg}
+            </span>
+          {case 'DELETE'}
+            <span class="delete">
+              {msg desc="Text for a deleted tree entry"}
+                [Deleted - <a href="{$entry.diffUrl}">diff</a>]
+              {/msg}
+            </span>
+          {case 'RENAME'}
+            <span class="rename">
+              {msg desc="Text for a renamed tree entry"}
+                [Renamed from {$entry.oldPath} - <a href="{$entry.diffUrl}">diff</a>]
+              {/msg}
+            </span>
+          {case 'COPY'}
+            <span class="copy">
+              {msg desc="Text for a copied tree entry"}
+                [Copied from {$entry.oldPath} - <a href="{$entry.diffUrl}">diff</a>]
+              {/msg}
+            </span>
+          {default}
+        {/switch}
+      </li>
+    {/foreach}
+  </ul>
+  <div class="diff-summary">
+    {if length($diffTree) == 1}
+      {msg desc="1 file changed"}1 file changed{/msg}
+    {else}
+      {msg desc="number of files changed"}{length($diffTree)} files changed{/msg}
+    {/if}
+  </div>
+{/if}
+
+{/template}
+
+/**
+ * Detailed listing of a tree.
+ *
+ * @param sha SHA of this path's tree.
+ * @param? logUrl optional URL to a log for this path.
+ * @param entries list of entries with the following keys:
+ *     type: entry type, matching one of the constant names defined in
+ *         org.eclipse.jgit.lib.FileMode.
+ *     name: tree entry name.
+ *     url: URL to link to.
+ *     targetName: name of a symlink target, required only if type == 'SYMLINK'.
+ *     targetUrl: optional url of a symlink target, required only if
+ *         type == 'SYMLINK'.
+ */
+{template .treeDetail}
+<div class="sha1">
+  {msg desc="SHA-1 for the path's tree"}tree: {$sha}{/msg}
+  {if $logUrl}{sp}[<a href="{$logUrl}">{msg desc="history for a path"}path history{/msg}</a>]{/if}
+</div>
+
+{if length($entries)}
+  <ol class="list files">
+    {foreach $entry in $entries}
+      <li class="
+            {switch $entry.type}
+              {case 'TREE'}git-tree
+              {case 'SYMLINK'}symlink
+              {case 'REGULAR_FILE'}regular-file
+              {case 'EXECUTABLE_FILE'}executable-file
+              {case 'GITLINK'}gitlink
+              {default}regular-file
+            {/switch}
+            " title="
+            {switch $entry.type}
+              {case 'TREE'}{msg desc="Alt text for tree icon"}Tree{/msg}
+              {case 'SYMLINK'}{msg desc="Alt text for symlink icon"}Symlink{/msg}
+              {case 'REGULAR_FILE'}{msg desc="Alt text for regular file icon"}Regular file{/msg}
+              {case 'EXECUTABLE_FILE'}
+                {msg desc="Alt text for executable file icon"}Executable file{/msg}
+              {case 'GITLINK'}
+                {msg desc="Alt text for git submodule link icon"}Git submodule link{/msg}
+              {default}{msg desc="Alt text for other file icon"}Other{/msg}
+            {/switch}
+            - {$entry.name}">
+        <a href="{$entry.url}">{$entry.name}</a>
+        {if $entry.type == 'SYMLINK'}
+          {sp}&#x21e8;{sp}
+          {if $entry.targetUrl}
+            <a href="{$entry.targetUrl}">{$entry.targetName}</a>
+          {else}
+            {$entry.targetName}
+          {/if}
+        {/if}
+        // TODO(dborowitz): Something reasonable for gitlinks.
+      </li>
+    {/foreach}
+  </table>
+{else}
+  <p>{msg desc="Informational text for when a tree is empty"}This tree is empty.{/msg}</p>
+{/if}
+{/template}
+
+/**
+ * Detailed listing of a blob.
+ *
+ * @param sha SHA of this file's blob.
+ * @param? logUrl optional URL to a log for this file.
+ * @param data file data (may be empty), or null for a binary file.
+ * @param? lang prettyprint language extension for text file.
+ * @param? size for binary files only, size in bytes.
+ */
+{template .blobDetail}
+<div class="sha1">
+  {msg desc="SHA-1 for the file's blob"}blob: {$sha}{/msg}
+  {if $logUrl}{sp}[<a href="{$logUrl}">{msg desc="history for a file"}file history{/msg}</a>]{/if}
+</div>
+
+{if $data != null}
+  {if $data}
+    {if $lang != null}
+      <pre class="git-blob prettyprint linenums lang-{$lang}">{$data}</pre>
+    {else}
+      <pre class="git-blob">{$data}</pre>
+    {/if}
+  {else}
+    <div class="file-empty">Empty file</div>
+  {/if}
+{else}
+  <div class="file-binary">
+    {msg desc="size of binary file in bytes"}{$size}-byte binary file{/msg}
+  </div>
+{/if}
+{/template}
+
+/**
+ * Detailed listing of an annotated tag.
+ *
+ * @param sha SHA of this tag.
+ * @param? tagger optional map with "name", "email", and "time" keys for the
+ *     tagger.
+ * @param object SHA of the object this tag points to.
+ * @param message tag message.
+ */
+{template .tagDetail}
+<div class="git-tag">
+  <table>
+    <tr>
+      <th>{msg desc="Header for tag SHA entry"}tag{/msg}</th>
+      <td class="sha">{$sha}</td>
+      <td>{sp}</td>
+    </tr>
+    {if $tagger}
+      <tr>
+        <th>{msg desc="Header for tagger"}tagger{/msg}</th>
+        <td>{call .person_ data="$tagger" /}</td>
+        <td>{$tagger.time}</td>
+      </tr>
+    {/if}
+    <tr>
+      <th>{msg desc="Header for tagged object SHA"}object{/msg}</th>
+      <td class="sha">{$object}</td>
+      <td>{sp}</td>
+    </tr>
+  </table>
+</div>
+{if $message and length($message)}
+  {call .message_}
+    {param className: 'tag-message' /}
+    {param message: $message /}
+  {/call}
+{/if}
+{/template}
+
+/**
+ * Line about a git person identity.
+ *
+ * @param name name.
+ * @param email email.
+ */
+{template .person_ private="true"}
+{$name}{if $email} &lt;{$email}&gt;{/if}
+{/template}
+
+/**
+ * Preformatted message, possibly containing hyperlinks.
+ *
+ * @param className CSS class name for <pre> block.
+ * @param message list of message parts, where each part contains:
+ *     text: raw text of the part.
+ *     url: optional URL that should be linked to from the part.
+ */
+{template .message_ private="true"}
+<pre class="{$className|id}">
+  {foreach $part in $message}
+    {if $part.url}<a href="{$part.url}">{$part.text}</a>{else}{$part.text}{/if}
+  {/foreach}
+</pre>
+{/template}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/PathDetail.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/PathDetail.soy
new file mode 100644
index 0000000..33d38a2
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/PathDetail.soy
@@ -0,0 +1,84 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * Detail page for a path within a tree.
+ *
+ * @param title human-readable name of this path.
+ * @param repositoryName name of this repository.
+ * @param? menuEntries menu entries.
+ * @param breadcrumbs breadcrumbs for this page.
+ * @param type path type, matching one of the constant names defined in
+ *         org.eclipse.jgit.lib.FileMode.
+ * @param data path data, matching the params for one of .treeDetail,
+ *     .blobDetail, .symlinkDetail, or .gitlinkDetail as appropriate.
+ */
+{template .pathDetail}
+{if $type == 'REGULAR_FILE' or $type == 'EXECUTABLE_FILE'}
+  {call .header}
+    {param title: $title /}
+    {param repositoryName: $repositoryName /}
+    {param menuEntries: $menuEntries /}
+    {param breadcrumbs: $breadcrumbs /}
+    {param css: [gitiles.PRETTIFY_CSS_URL] /}
+    {param js: [gitiles.PRETTIFY_JS_URL] /}
+    {param onLoad: 'prettyPrint()' /}
+  {/call}
+{else}
+  {call .header data="all" /}
+{/if}
+
+{switch $type}
+  {case 'TREE'}{call .treeDetail data="$data" /}
+  {case 'SYMLINK'}{call .symlinkDetail data="$data" /}
+  {case 'REGULAR_FILE'}{call .blobDetail data="$data" /}
+  {case 'EXECUTABLE_FILE'}{call .blobDetail data="$data" /}
+  {case 'GITLINK'}{call .gitlinkDetail data="$data" /}
+  {default}
+    <div class="error">
+      {msg desc="Error message for an unknown object type"}Object has unknown type.{/msg}
+    </div>
+{/switch}
+
+{call .footer /}
+{/template}
+
+/**
+ * Detail for a symbolic link.
+ *
+ * @param target target of this symlink.
+ * @param? targetUrl optional URL for the target, if it is within this repo.
+ */
+{template .symlinkDetail}
+<div class="symlink-detail">
+  {msg desc="Lead-in text for the symbolic link target."}Symbolic link to{/msg}
+  {sp}{if $targetUrl}<a href="{$targetUrl}">{$target}</a>{else}{$target}{/if}
+</div>
+{/template}
+
+/**
+ * Detail for a git submodule link.
+ *
+ * @param sha submodule commit SHA.
+ * @param remoteUrl URL of the remote repository.
+ * @param? httpUrl optional HTTP URL pointing to a web-browser-compatible URL of
+ *     the remote repository.
+ */
+{template .gitlinkDetail}
+<div class="gitlink-detail">
+  {msg desc="Lead-in text for the git link URL"}Submodule link to {$sha} of{/msg}
+  {sp}{if $httpUrl}<a href="{$httpUrl}">{$remoteUrl}</a>{else}{$remoteUrl}{/if}
+</div>
+{/template}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/RepositoryIndex.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/RepositoryIndex.soy
new file mode 100644
index 0000000..02bb4da
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/RepositoryIndex.soy
@@ -0,0 +1,79 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * Index page for a repository.
+ *
+ * @param repositoryName name of this repository.
+ * @param? menuEntries menu entries.
+ * @param breadcrumbs breadcrumbs for this page.
+ * @param cloneUrl clone URL for this repository.
+ * @param description description text of the repository.
+ * @param? branches list of branch objects with url and name keys.
+ * @param? tags list of tag objects with url and name keys.
+ */
+{template .repositoryIndex}
+{call .header}
+  {param title: $repositoryName /}
+  {param repositoryName: null /}
+  {param menuEntries: $menuEntries /}
+  {param breadcrumbs: $breadcrumbs /}
+{/call}
+
+{if $description}
+  <div class="repository-description">{$description}</div>
+{/if}
+
+<textarea rows="1" cols="150" class="clone-line"
+  onclick="this.focus();this.select();"
+  readonly="readonly">
+    git clone {$cloneUrl}
+</textarea>
+
+<div class="repository-refs">
+  {if $branches and length($branches)}
+    <div class="repository-branches">
+      <h3>Branches</h3>
+      <ul class="branch-list">
+      {foreach $branch in $branches}
+        {call .ref_ data="$branch" /}
+      {/foreach}
+      </ul>
+    </div>
+  {/if}
+
+  {if $tags and length($tags)}
+    <div class="repository-tags">
+      <h3>Tags</h3>
+      <ul class="branch-list">
+      {foreach $tag in $tags}
+        {call .ref_ data="$tag" /}
+      {/foreach}
+      </ul>
+    </div>
+  {/if}
+</div>
+{call .footer /}
+{/template}
+
+/**
+ * Detail for a single ref.
+ *
+ * @param url URL for ref detail page.
+ * @param name ref name.
+ */
+{template .ref_}
+<li><a href="{$url}">{$name}</a></li>
+{/template}
diff --git a/gitiles-servlet/src/main/resources/com/google/gitiles/templates/RevisionDetail.soy b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/RevisionDetail.soy
new file mode 100644
index 0000000..1abed6b
--- /dev/null
+++ b/gitiles-servlet/src/main/resources/com/google/gitiles/templates/RevisionDetail.soy
@@ -0,0 +1,63 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.
+{namespace gitiles autoescape="contextual"}
+
+/**
+ * Detail page about a single revision.
+ *
+ * @param title human-readable revision name.
+ * @param repositoryName name of this repository.
+ * @param? menuEntries menu entries.
+ * @param breadcrumbs breadcrumbs for this page.
+ * @param? hasBlob set to true if the revision or its peeled value is a blob.
+ * @param objects list of objects encountered when peeling this object. Each
+ *     object has a "type" key with one of the
+ *     org.eclipse.jgit.lib.Contants.TYPE_* constant strings, and a "data" key
+ *     with an object whose keys correspond to the appropriate object detail
+ *     template from ObjectDetail.soy.
+ */
+{template .revisionDetail}
+{if $hasBlob}
+  {call .header}
+    {param title: $title /}
+    {param repositoryName: $repositoryName /}
+    {param menuEntries: $menuEntries /}
+    {param breadcrumbs: $breadcrumbs /}
+    {param css: [gitiles.PRETTIFY_CSS_URL] /}
+    {param js: [gitiles.PRETTIFY_JS_URL] /}
+    {param onLoad: 'prettyPrint()' /}
+  {/call}
+{else}
+  {call .header data="all" /}
+{/if}
+
+{foreach $object in $objects}
+  {switch $object.type}
+    {case 'commit'}
+      {call .commitDetail data="$object.data" /}
+    {case 'tree'}
+      {call .treeDetail data="$object.data" /}
+    {case 'blob'}
+      {call .blobDetail data="$object.data" /}
+    {case 'tag'}
+      {call .tagDetail data="$object.data" /}
+    {default}
+      <div class="error">
+        {msg desc="Error message for an unknown object type"}Object has unknown type.{/msg}
+      </div>
+  {/switch}
+{/foreach}
+
+{call .footer /}
+{/template}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/ConfigUtilTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/ConfigUtilTest.java
new file mode 100644
index 0000000..96439c2
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/ConfigUtilTest.java
@@ -0,0 +1,43 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.gitiles.ConfigUtil.getDuration;
+
+import junit.framework.TestCase;
+
+import org.eclipse.jgit.lib.Config;
+import org.joda.time.Duration;
+
+/** Tests for configuration utilities. */
+public class ConfigUtilTest extends TestCase {
+  public void testGetDuration() throws Exception {
+    Duration def = Duration.standardSeconds(2);
+    Config config = new Config();
+    Duration t;
+
+    config.setString("core", "dht", "timeout", "500 ms");
+    t = getDuration(config, "core", "dht", "timeout", def);
+    assertEquals(500, t.getMillis());
+
+    config.setString("core", "dht", "timeout", "5.2 sec");
+    t = getDuration(config, "core", "dht", "timeout", def);
+    assertEquals(5200, t.getMillis());
+
+    config.setString("core", "dht", "timeout", "1 min");
+    t = getDuration(config, "core", "dht", "timeout", def);
+    assertEquals(60000, t.getMillis());
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/FakeHttpServletRequest.java b/gitiles-servlet/src/test/java/com/google/gitiles/FakeHttpServletRequest.java
new file mode 100644
index 0000000..c1afa74
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/FakeHttpServletRequest.java
@@ -0,0 +1,389 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Charsets.UTF_8;
+import static com.google.common.base.Preconditions.checkArgument;
+import static com.google.common.base.Preconditions.checkNotNull;
+import static com.google.gitiles.TestGitilesUrls.URLS;
+
+import com.google.common.base.Function;
+import com.google.common.base.Splitter;
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.Iterables;
+import com.google.common.collect.LinkedListMultimap;
+import com.google.common.collect.ListMultimap;
+import com.google.common.collect.Maps;
+
+import org.eclipse.jgit.http.server.ServletUtils;
+import org.eclipse.jgit.storage.dfs.DfsRepository;
+
+import java.io.BufferedReader;
+import java.io.UnsupportedEncodingException;
+import java.net.URLDecoder;
+import java.security.Principal;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.Enumeration;
+import java.util.List;
+import java.util.Locale;
+import java.util.Map;
+
+import javax.servlet.RequestDispatcher;
+import javax.servlet.ServletInputStream;
+import javax.servlet.http.Cookie;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpSession;
+
+/** Simple fake implementation of {@link HttpServletRequest}. */
+public class FakeHttpServletRequest implements HttpServletRequest {
+  public static final String SERVLET_PATH = "/b";
+
+  public static FakeHttpServletRequest newRequest() {
+    return new FakeHttpServletRequest(
+        URLS.getHostName(null),
+        80,
+        "",
+        SERVLET_PATH,
+        "");
+  }
+
+  public static FakeHttpServletRequest newRequest(DfsRepository repo) {
+    FakeHttpServletRequest req = newRequest();
+    req.setAttribute(ServletUtils.ATTRIBUTE_REPOSITORY, repo);
+    return req;
+  }
+
+  private final Map<String, Object> attributes;
+  private final ListMultimap<String, String> headers;
+
+  private ListMultimap<String, String> parameters;
+  private String hostName;
+  private int port;
+  private String contextPath;
+  private String servletPath;
+  private String path;
+
+  private FakeHttpServletRequest(String hostName, int port, String contextPath, String servletPath,
+      String path) {
+    this.hostName = checkNotNull(hostName, "hostName");
+    checkArgument(port > 0);
+    this.port = port;
+    this.contextPath = checkNotNull(contextPath, "contextPath");
+    this.servletPath = checkNotNull(servletPath, "servletPath");
+    attributes = Maps.newConcurrentMap();
+    parameters = LinkedListMultimap.create();
+    headers = LinkedListMultimap.create();
+  }
+
+  @Override
+  public Object getAttribute(String name) {
+    return attributes.get(name);
+  }
+
+  @Override
+  public Enumeration<String> getAttributeNames() {
+    return Collections.enumeration(attributes.keySet());
+  }
+
+  @Override
+  public String getCharacterEncoding() {
+    return UTF_8.name();
+  }
+
+  @Override
+  public int getContentLength() {
+    return -1;
+  }
+
+  @Override
+  public String getContentType() {
+    return null;
+  }
+
+  @Override
+  public ServletInputStream getInputStream() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public String getLocalAddr() {
+    return "1.2.3.4";
+  }
+
+  @Override
+  public String getLocalName() {
+    return hostName;
+  }
+
+  @Override
+  public int getLocalPort() {
+    return port;
+  }
+
+  @Override
+  public Locale getLocale() {
+    return Locale.US;
+  }
+
+  @Override
+  public Enumeration<Locale> getLocales() {
+    return Collections.enumeration(Collections.singleton(Locale.US));
+  }
+
+  @Override
+  public String getParameter(String name) {
+    return Iterables.getFirst(parameters.get(name), null);
+  }
+
+  private static final Function<Collection<String>, String[]> STRING_COLLECTION_TO_ARRAY =
+      new Function<Collection<String>, String[]>() {
+        @Override
+        public String[] apply(Collection<String> values) {
+          return values.toArray(new String[0]);
+        }
+      };
+
+  @Override
+  public Map<String, String[]> getParameterMap() {
+    return Collections.unmodifiableMap(
+        Maps.transformValues(parameters.asMap(), STRING_COLLECTION_TO_ARRAY));
+  }
+
+  @Override
+  public Enumeration<String> getParameterNames() {
+    return Collections.enumeration(parameters.keySet());
+  }
+
+  @Override
+  public String[] getParameterValues(String name) {
+    return STRING_COLLECTION_TO_ARRAY.apply(parameters.get(name));
+  }
+
+  public void setQueryString(String qs) {
+    ListMultimap<String, String> params = LinkedListMultimap.create();
+    for (String entry : Splitter.on('&').split(qs)) {
+      List<String> kv = ImmutableList.copyOf(Splitter.on('=').limit(2).split(entry));
+      try {
+        params.put(URLDecoder.decode(kv.get(0), UTF_8.name()),
+            kv.size() == 2 ? URLDecoder.decode(kv.get(1), UTF_8.name()) : "");
+      } catch (UnsupportedEncodingException e) {
+        throw new IllegalArgumentException(e);
+      }
+    }
+    parameters = params;
+  }
+
+  @Override
+  public String getProtocol() {
+    return "HTTP/1.1";
+  }
+
+  @Override
+  public BufferedReader getReader() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  @Deprecated
+  public String getRealPath(String path) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public String getRemoteAddr() {
+    return "5.6.7.8";
+  }
+
+  @Override
+  public String getRemoteHost() {
+    return "remotehost";
+  }
+
+  @Override
+  public int getRemotePort() {
+    return 1234;
+  }
+
+  @Override
+  public RequestDispatcher getRequestDispatcher(String path) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public String getScheme() {
+    return port == 443 ? "https" : "http";
+  }
+
+  @Override
+  public String getServerName() {
+    return hostName;
+  }
+
+  @Override
+  public int getServerPort() {
+    return port;
+  }
+
+  @Override
+  public boolean isSecure() {
+    return port == 443;
+  }
+
+  @Override
+  public void removeAttribute(String name) {
+    attributes.remove(name);
+  }
+
+  @Override
+  public void setAttribute(String name, Object value) {
+    attributes.put(name, value);
+  }
+
+  @Override
+  public void setCharacterEncoding(String env) throws UnsupportedOperationException {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public String getAuthType() {
+    return null;
+  }
+
+  @Override
+  public String getContextPath() {
+    return contextPath;
+  }
+
+  @Override
+  public Cookie[] getCookies() {
+    return new Cookie[0];
+  }
+
+  @Override
+  public long getDateHeader(String name) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public String getHeader(String name) {
+    return Iterables.getFirst(headers.get(name), null);
+  }
+
+  @Override
+  public Enumeration<String> getHeaderNames() {
+    return Collections.enumeration(headers.keySet());
+  }
+
+  @Override
+  public Enumeration<String> getHeaders(String name) {
+    return Collections.enumeration(headers.get(name));
+  }
+
+  @Override
+  public int getIntHeader(String name) {
+    return Integer.parseInt(getHeader(name));
+  }
+
+  @Override
+  public String getMethod() {
+    return "GET";
+  }
+
+  @Override
+  public String getPathInfo() {
+    return path;
+  }
+
+  public void setPathInfo(String path) {
+    this.path = checkNotNull(path);
+  }
+
+  @Override
+  public String getPathTranslated() {
+    return path;
+  }
+
+  @Override
+  public String getQueryString() {
+    return null;
+  }
+
+  @Override
+  public String getRemoteUser() {
+    return null;
+  }
+
+  @Override
+  public String getRequestURI() {
+    return null;
+  }
+
+  @Override
+  public StringBuffer getRequestURL() {
+    return null;
+  }
+
+  @Override
+  public String getRequestedSessionId() {
+    return null;
+  }
+
+  @Override
+  public String getServletPath() {
+    return servletPath;
+  }
+
+  @Override
+  public HttpSession getSession() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public HttpSession getSession(boolean create) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public Principal getUserPrincipal() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public boolean isRequestedSessionIdFromCookie() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public boolean isRequestedSessionIdFromURL() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  @Deprecated
+  public boolean isRequestedSessionIdFromUrl() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public boolean isRequestedSessionIdValid() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public boolean isUserInRole(String role) {
+    throw new UnsupportedOperationException();
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/FakeHttpServletResponse.java b/gitiles-servlet/src/test/java/com/google/gitiles/FakeHttpServletResponse.java
new file mode 100644
index 0000000..87a0099
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/FakeHttpServletResponse.java
@@ -0,0 +1,201 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Charsets.UTF_8;
+
+import java.io.PrintWriter;
+import java.util.Locale;
+
+import javax.servlet.ServletOutputStream;
+import javax.servlet.http.Cookie;
+import javax.servlet.http.HttpServletResponse;
+
+/** Simple fake implementation of {@link HttpServletResponse}. */
+public class FakeHttpServletResponse implements HttpServletResponse {
+
+  private volatile int status;
+
+  public FakeHttpServletResponse() {
+    status = 200;
+  }
+
+  @Override
+  public void flushBuffer() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public int getBufferSize() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public String getCharacterEncoding() {
+    return UTF_8.name();
+  }
+
+  @Override
+  public String getContentType() {
+    return null;
+  }
+
+  @Override
+  public Locale getLocale() {
+    return Locale.US;
+  }
+
+  @Override
+  public ServletOutputStream getOutputStream() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public PrintWriter getWriter() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public boolean isCommitted() {
+    return false;
+  }
+
+  @Override
+  public void reset() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void resetBuffer() {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void setBufferSize(int sz) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void setCharacterEncoding(String name) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void setContentLength(int length) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void setContentType(String type) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void setLocale(Locale locale) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void addCookie(Cookie cookie) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void addDateHeader(String name, long value) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void addHeader(String name, String value) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void addIntHeader(String name, int value) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public boolean containsHeader(String name) {
+    return false;
+  }
+
+  @Override
+  public String encodeRedirectURL(String url) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  @Deprecated
+  public String encodeRedirectUrl(String url) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public String encodeURL(String url) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  @Deprecated
+  public String encodeUrl(String url) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void sendError(int sc) {
+    status = sc;
+  }
+
+  @Override
+  public void sendError(int sc, String msg) {
+    status = sc;
+  }
+
+  @Override
+  public void sendRedirect(String msg) {
+    status = SC_FOUND;
+  }
+
+  @Override
+  public void setDateHeader(String name, long value) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void setHeader(String name, String value) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void setIntHeader(String name, int value) {
+    throw new UnsupportedOperationException();
+  }
+
+  @Override
+  public void setStatus(int sc) {
+    status = sc;
+  }
+
+  @Override
+  @Deprecated
+  public void setStatus(int sc, String msg) {
+    status = sc;
+  }
+
+  public int getStatus() {
+    return status;
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/GitilesFilterTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/GitilesFilterTest.java
new file mode 100644
index 0000000..0d32dd4
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/GitilesFilterTest.java
@@ -0,0 +1,171 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.gitiles.GitilesFilter.REPO_PATH_REGEX;
+import static com.google.gitiles.GitilesFilter.REPO_REGEX;
+import static com.google.gitiles.GitilesFilter.ROOT_REGEX;
+
+import junit.framework.TestCase;
+
+import java.util.regex.Matcher;
+
+/** Tests for the Gitiles filter. */
+public class GitilesFilterTest extends TestCase {
+  public void testRootUrls() throws Exception {
+    assertFalse(ROOT_REGEX.matcher("").matches());
+    assertFalse(ROOT_REGEX.matcher("/foo").matches());
+    assertFalse(ROOT_REGEX.matcher("/foo/").matches());
+    assertFalse(ROOT_REGEX.matcher("/foo/ ").matches());
+    assertFalse(ROOT_REGEX.matcher("/foo/+").matches());
+    assertFalse(ROOT_REGEX.matcher("/foo/+").matches());
+    assertFalse(ROOT_REGEX.matcher("/foo/ /").matches());
+    assertFalse(ROOT_REGEX.matcher("/foo/+/").matches());
+    assertFalse(ROOT_REGEX.matcher("/foo/+/bar").matches());
+    Matcher m;
+
+    m = ROOT_REGEX.matcher("/");
+    assertTrue(m.matches());
+    assertEquals("/", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/", m.group(2));
+    assertEquals("", m.group(3));
+    assertEquals("", m.group(4));
+
+    m = ROOT_REGEX.matcher("//");
+    assertTrue(m.matches());
+    assertEquals("//", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/", m.group(2));
+    assertEquals("", m.group(3));
+    assertEquals("", m.group(4));
+  }
+
+  public void testRepoUrls() throws Exception {
+    assertFalse(REPO_REGEX.matcher("").matches());
+
+    // These match the regex but are served by the root regex binder, which is
+    // matched first.
+    assertTrue(REPO_REGEX.matcher("/").matches());
+    assertTrue(REPO_REGEX.matcher("//").matches());
+
+    assertFalse(REPO_REGEX.matcher("/foo/+").matches());
+    assertFalse(REPO_REGEX.matcher("/foo/bar/+").matches());
+    assertFalse(REPO_REGEX.matcher("/foo/bar/+/").matches());
+    assertFalse(REPO_REGEX.matcher("/foo/bar/+/baz").matches());
+    Matcher m;
+
+    m = REPO_REGEX.matcher("/foo");
+    assertTrue(m.matches());
+    assertEquals("/foo", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("", m.group(3));
+    assertEquals("", m.group(4));
+
+    m = REPO_REGEX.matcher("/foo/");
+    assertTrue(m.matches());
+    assertEquals("/foo/", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("", m.group(3));
+    assertEquals("", m.group(4));
+
+    m = REPO_REGEX.matcher("/foo/bar");
+    assertTrue(m.matches());
+    assertEquals("/foo/bar", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo/bar", m.group(2));
+    assertEquals("", m.group(3));
+    assertEquals("", m.group(4));
+
+    m = REPO_REGEX.matcher("/foo/bar+baz");
+    assertTrue(m.matches());
+    assertEquals("/foo/bar+baz", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo/bar+baz", m.group(2));
+    assertEquals("", m.group(3));
+    assertEquals("", m.group(4));
+  }
+
+  public void testRepoPathUrls() throws Exception {
+    assertFalse(REPO_PATH_REGEX.matcher("").matches());
+    assertFalse(REPO_PATH_REGEX.matcher("/").matches());
+    assertFalse(REPO_PATH_REGEX.matcher("//").matches());
+    assertFalse(REPO_PATH_REGEX.matcher("/foo").matches());
+    assertFalse(REPO_PATH_REGEX.matcher("/foo/ ").matches());
+    assertFalse(REPO_PATH_REGEX.matcher("/foo/ /").matches());
+    assertFalse(REPO_PATH_REGEX.matcher("/foo/ /bar").matches());
+    assertFalse(REPO_PATH_REGEX.matcher("/foo/bar").matches());
+    assertFalse(REPO_PATH_REGEX.matcher("/foo/bar+baz").matches());
+    Matcher m;
+
+    m = REPO_PATH_REGEX.matcher("/foo/+");
+    assertTrue(m.matches());
+    assertEquals("/foo/+", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("+", m.group(3));
+    assertEquals("", m.group(4));
+
+    m = REPO_PATH_REGEX.matcher("/foo/+/");
+    assertTrue(m.matches());
+    assertEquals("/foo/+/", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("+", m.group(3));
+    assertEquals("/", m.group(4));
+
+    m = REPO_PATH_REGEX.matcher("/foo/+/bar/baz");
+    assertTrue(m.matches());
+    assertEquals("/foo/+/bar/baz", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("+", m.group(3));
+    assertEquals("/bar/baz", m.group(4));
+
+    m = REPO_PATH_REGEX.matcher("/foo/+/bar/baz/");
+    assertTrue(m.matches());
+    assertEquals("/foo/+/bar/baz/", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("+", m.group(3));
+    assertEquals("/bar/baz/", m.group(4));
+
+    m = REPO_PATH_REGEX.matcher("/foo/+/bar baz");
+    assertTrue(m.matches());
+    assertEquals("/foo/+/bar baz", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("+", m.group(3));
+    assertEquals("/bar baz", m.group(4));
+
+    m = REPO_PATH_REGEX.matcher("/foo/+/bar/+/baz");
+    assertTrue(m.matches());
+    assertEquals("/foo/+/bar/+/baz", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("+", m.group(3));
+    assertEquals("/bar/+/baz", m.group(4));
+
+    m = REPO_PATH_REGEX.matcher("/foo/+bar/baz");
+    assertTrue(m.matches());
+    assertEquals("/foo/+bar/baz", m.group(0));
+    assertEquals(m.group(0), m.group(1));
+    assertEquals("/foo", m.group(2));
+    assertEquals("+bar", m.group(3));
+    assertEquals("/baz", m.group(4));
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/GitilesUrlsTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/GitilesUrlsTest.java
new file mode 100644
index 0000000..4c2ff8b
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/GitilesUrlsTest.java
@@ -0,0 +1,48 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.gitiles.GitilesUrls.NAME_ESCAPER;
+
+import junit.framework.TestCase;
+
+/** Unit tests for {@link GitilesUrls}. */
+public class GitilesUrlsTest extends TestCase {
+  public void testNameEscaperEscapesAppropriateSpecialCharacters() throws Exception {
+    assertEquals("foo_bar", NAME_ESCAPER.apply("foo_bar"));
+    assertEquals("foo-bar", NAME_ESCAPER.apply("foo-bar"));
+    assertEquals("foo%25bar", NAME_ESCAPER.apply("foo%bar"));
+    assertEquals("foo%26bar", NAME_ESCAPER.apply("foo&bar"));
+    assertEquals("foo%28bar", NAME_ESCAPER.apply("foo(bar"));
+    assertEquals("foo%29bar", NAME_ESCAPER.apply("foo)bar"));
+    assertEquals("foo%3Abar", NAME_ESCAPER.apply("foo:bar"));
+    assertEquals("foo%3Bbar", NAME_ESCAPER.apply("foo;bar"));
+    assertEquals("foo%3Dbar", NAME_ESCAPER.apply("foo=bar"));
+    assertEquals("foo%3Fbar", NAME_ESCAPER.apply("foo?bar"));
+    assertEquals("foo%5Bbar", NAME_ESCAPER.apply("foo[bar"));
+    assertEquals("foo%5Dbar", NAME_ESCAPER.apply("foo]bar"));
+    assertEquals("foo%7Bbar", NAME_ESCAPER.apply("foo{bar"));
+    assertEquals("foo%7Dbar", NAME_ESCAPER.apply("foo}bar"));
+  }
+  public void testNameEscaperDoesNotEscapeSlashes() throws Exception {
+    assertEquals("foo/bar", NAME_ESCAPER.apply("foo/bar"));
+  }
+
+  public void testNameEscaperEscapesSpacesWithPercentInsteadOfPlus() throws Exception {
+    assertEquals("foo+bar", NAME_ESCAPER.apply("foo+bar"));
+    assertEquals("foo%20bar", NAME_ESCAPER.apply("foo bar"));
+    assertEquals("foo%2520bar", NAME_ESCAPER.apply("foo%20bar"));
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/GitilesViewTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/GitilesViewTest.java
new file mode 100644
index 0000000..a8a5c4f
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/GitilesViewTest.java
@@ -0,0 +1,504 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableListMultimap;
+import com.google.common.collect.ImmutableMap;
+import com.google.gitiles.GitilesView.Type;
+
+import junit.framework.TestCase;
+
+import org.eclipse.jgit.lib.ObjectId;
+
+/** Tests for Gitiles views. */
+public class GitilesViewTest extends TestCase {
+  private static final GitilesView HOST = GitilesView.hostIndex()
+      .setServletPath("/b")
+      .setHostName("host")
+      .build();
+
+  public void testEmptyServletPath() throws Exception {
+    GitilesView view = GitilesView.hostIndex()
+        .setServletPath("")
+        .setHostName("host")
+        .build();
+    assertEquals("", view.getServletPath());
+    assertEquals(Type.HOST_INDEX, view.getType());
+    assertEquals("host", view.getHostName());
+    assertNull(view.getRepositoryName());
+    assertEquals(Revision.NULL, view.getRevision());
+    assertNull(view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/?format=HTML", view.toUrl());
+    assertEquals(ImmutableList.of(ImmutableMap.of("text", "host", "url", "/?format=HTML")),
+        view.getBreadcrumbs());
+  }
+
+  public void testHostIndex() throws Exception {
+    assertEquals("/b", HOST.getServletPath());
+    assertEquals(Type.HOST_INDEX, HOST.getType());
+    assertEquals("host", HOST.getHostName());
+    assertNull(HOST.getRepositoryName());
+    assertEquals(Revision.NULL, HOST.getRevision());
+    assertNull(HOST.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/?format=HTML", HOST.toUrl());
+    assertEquals(ImmutableList.of(ImmutableMap.of("text", "host", "url", "/b/?format=HTML")),
+        HOST.getBreadcrumbs());
+  }
+
+  public void testQueryParams() throws Exception {
+    GitilesView view = GitilesView.hostIndex().copyFrom(HOST)
+        .putParam("foo", "foovalue")
+        .putParam("bar", "barvalue")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.HOST_INDEX, view.getType());
+    assertEquals("host", view.getHostName());
+    assertNull(view.getRepositoryName());
+    assertEquals(Revision.NULL, view.getRevision());
+    assertNull(view.getTreePath());
+    assertEquals(
+        ImmutableListMultimap.of(
+            "foo", "foovalue",
+            "bar", "barvalue"),
+        view.getParameters());
+
+    assertEquals("/b/?format=HTML&foo=foovalue&bar=barvalue", view.toUrl());
+    assertEquals(ImmutableList.of(ImmutableMap.of("text", "host", "url", "/b/?format=HTML")),
+        view.getBreadcrumbs());
+  }
+
+  public void testQueryParamsNotCopied() throws Exception {
+    GitilesView view = GitilesView.hostIndex().copyFrom(HOST)
+        .putParam("foo", "foovalue")
+        .putParam("bar", "barvalue")
+        .build();
+    GitilesView copy = GitilesView.hostIndex().copyFrom(view).build();
+    assertFalse(view.getParameters().isEmpty());
+    assertTrue(copy.getParameters().isEmpty());
+  }
+
+  public void testRepositoryIndex() throws Exception {
+    GitilesView view = GitilesView.repositoryIndex()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.REPOSITORY_INDEX, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(Revision.NULL, view.getRevision());
+    assertNull(view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/")),
+        view.getBreadcrumbs());
+  }
+
+  public void testRefWithRevision() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.revision()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertNull(view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+show/master", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master", "/b/foo/bar/+show/master")),
+        view.getBreadcrumbs());
+  }
+
+  public void testNoPathComponents() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.path()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .setTreePath("/")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.PATH, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals("", view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+/master/", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master", "/b/foo/bar/+show/master"),
+            breadcrumb(".", "/b/foo/bar/+/master/")),
+        view.getBreadcrumbs());
+  }
+
+  public void testOnePathComponent() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.path()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .setTreePath("/file")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.PATH, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals("file", view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+/master/file", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master", "/b/foo/bar/+show/master"),
+            breadcrumb(".", "/b/foo/bar/+/master/"),
+            breadcrumb("file", "/b/foo/bar/+/master/file")),
+        view.getBreadcrumbs());
+  }
+
+  public void testMultiplePathComponents() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.path()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .setTreePath("/path/to/a/file")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.PATH, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals("path/to/a/file", view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+/master/path/to/a/file", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master", "/b/foo/bar/+show/master"),
+            breadcrumb(".", "/b/foo/bar/+/master/"),
+            breadcrumb("path", "/b/foo/bar/+/master/path"),
+            breadcrumb("to", "/b/foo/bar/+/master/path/to"),
+            breadcrumb("a", "/b/foo/bar/+/master/path/to/a"),
+            breadcrumb("file", "/b/foo/bar/+/master/path/to/a/file")),
+        view.getBreadcrumbs());
+  }
+
+  public void testDiffAgainstFirstParent() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    ObjectId parent = ObjectId.fromString("efab5678efab5678efab5678efab5678efab5678");
+    GitilesView view = GitilesView.diff()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .setOldRevision(Revision.unpeeled("master^", parent))
+        .setTreePath("/path/to/a/file")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.DIFF, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals("master^", view.getOldRevision().getName());
+    assertEquals("path/to/a/file", view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+/master%5E%21/path/to/a/file", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master^!", "/b/foo/bar/+/master%5E%21/"),
+            breadcrumb(".", "/b/foo/bar/+/master%5E%21/"),
+            breadcrumb("path", "/b/foo/bar/+/master%5E%21/path"),
+            breadcrumb("to", "/b/foo/bar/+/master%5E%21/path/to"),
+            breadcrumb("a", "/b/foo/bar/+/master%5E%21/path/to/a"),
+            breadcrumb("file", "/b/foo/bar/+/master%5E%21/path/to/a/file")),
+        view.getBreadcrumbs());
+  }
+
+  public void testDiffAgainstEmptyRevision() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.diff()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .setTreePath("/path/to/a/file")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.DIFF, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertEquals("path/to/a/file", view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+/master%5E%21/path/to/a/file", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master^!", "/b/foo/bar/+/master%5E%21/"),
+            breadcrumb(".", "/b/foo/bar/+/master%5E%21/"),
+            breadcrumb("path", "/b/foo/bar/+/master%5E%21/path"),
+            breadcrumb("to", "/b/foo/bar/+/master%5E%21/path/to"),
+            breadcrumb("a", "/b/foo/bar/+/master%5E%21/path/to/a"),
+            breadcrumb("file", "/b/foo/bar/+/master%5E%21/path/to/a/file")),
+        view.getBreadcrumbs());
+  }
+
+  public void testDiffAgainstOther() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    ObjectId other = ObjectId.fromString("efab5678efab5678efab5678efab5678efab5678");
+    GitilesView view = GitilesView.diff()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .setOldRevision(Revision.unpeeled("efab5678", other))
+        .setTreePath("/path/to/a/file")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.DIFF, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals("efab5678", view.getOldRevision().getName());
+    assertEquals("path/to/a/file", view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+/efab5678..master/path/to/a/file", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("efab5678..master", "/b/foo/bar/+/efab5678..master/"),
+            breadcrumb(".", "/b/foo/bar/+/efab5678..master/"),
+            breadcrumb("path", "/b/foo/bar/+/efab5678..master/path"),
+            breadcrumb("to", "/b/foo/bar/+/efab5678..master/path/to"),
+            breadcrumb("a", "/b/foo/bar/+/efab5678..master/path/to/a"),
+            breadcrumb("file", "/b/foo/bar/+/efab5678..master/path/to/a/file")),
+        view.getBreadcrumbs());
+  }
+
+  public void testBranchLogWithoutPath() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.log()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertNull(view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+/master", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master", "/b/foo/bar/+/master")),
+        view.getBreadcrumbs());
+  }
+
+  public void testIdLogWithoutPath() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.log()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("abcd1234", id))
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("abcd1234", view.getRevision().getName());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertNull(view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+log/abcd1234", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("abcd1234", "/b/foo/bar/+log/abcd1234")),
+        view.getBreadcrumbs());
+  }
+
+  public void testLogWithoutOldRevision() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.log()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .setTreePath("/path/to/a/file")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertEquals("path/to/a/file", view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+log/master/path/to/a/file", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master", "/b/foo/bar/+/master"),
+            breadcrumb("path", "/b/foo/bar/+log/master/path"),
+            breadcrumb("to", "/b/foo/bar/+log/master/path/to"),
+            breadcrumb("a", "/b/foo/bar/+log/master/path/to/a"),
+            breadcrumb("file", "/b/foo/bar/+log/master/path/to/a/file")),
+        view.getBreadcrumbs());
+  }
+
+  public void testLogWithOldRevision() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    ObjectId parent = ObjectId.fromString("efab5678efab5678efab5678efab5678efab5678");
+    GitilesView view = GitilesView.log()
+        .copyFrom(HOST)
+        .setRepositoryName("foo/bar")
+        .setRevision(Revision.unpeeled("master", id))
+        .setOldRevision(Revision.unpeeled("master^", parent))
+        .setTreePath("/path/to/a/file")
+        .build();
+
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo/bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals("master^", view.getOldRevision().getName());
+    assertEquals("path/to/a/file", view.getTreePath());
+    assertTrue(HOST.getParameters().isEmpty());
+
+    assertEquals("/b/foo/bar/+log/master%5E..master/path/to/a/file", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo/bar", "/b/foo/bar/"),
+            breadcrumb("master^..master", "/b/foo/bar/+log/master%5E..master"),
+            breadcrumb("path", "/b/foo/bar/+log/master%5E..master/path"),
+            breadcrumb("to", "/b/foo/bar/+log/master%5E..master/path/to"),
+            breadcrumb("a", "/b/foo/bar/+log/master%5E..master/path/to/a"),
+            breadcrumb("file", "/b/foo/bar/+log/master%5E..master/path/to/a/file")),
+        view.getBreadcrumbs());
+  }
+
+  public void testEscaping() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    ObjectId parent = ObjectId.fromString("efab5678efab5678efab5678efab5678efab5678");
+    // Some of these values are not valid for Git, but check them anyway.
+    GitilesView view = GitilesView.log()
+        .copyFrom(HOST)
+        .setRepositoryName("foo?bar")
+        .setRevision(Revision.unpeeled("ba/d#name", id))
+        .setOldRevision(Revision.unpeeled("other\"na/me", parent))
+        .setTreePath("we ird/pa'th/name")
+        .putParam("k e y", "val/ue")
+        .setAnchor("anc#hor")
+        .build();
+
+    // Fields returned by getters are not escaped.
+    assertEquals("/b", view.getServletPath());
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("host", view.getHostName());
+    assertEquals("foo?bar", view.getRepositoryName());
+    assertEquals(id, view.getRevision().getId());
+    assertEquals("ba/d#name", view.getRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("other\"na/me", view.getOldRevision().getName());
+    assertEquals("we ird/pa'th/name", view.getTreePath());
+    assertEquals(ImmutableListMultimap.<String, String> of("k e y", "val/ue"),
+        view.getParameters());
+
+    assertEquals(
+        "/b/foo%3Fbar/+log/other%22na/me..ba/d%23name/we%20ird/pa%27th/name"
+        + "?k+e+y=val%2Fue#anc%23hor", view.toUrl());
+    assertEquals(
+        ImmutableList.of(
+            // Names are not escaped (auto-escaped by Soy) but values are.
+            breadcrumb("host", "/b/?format=HTML"),
+            breadcrumb("foo?bar", "/b/foo%3Fbar/"),
+            breadcrumb("other\"na/me..ba/d#name", "/b/foo%3Fbar/+log/other%22na/me..ba/d%23name"),
+            breadcrumb("we ird", "/b/foo%3Fbar/+log/other%22na/me..ba/d%23name/we%20ird"),
+            breadcrumb("pa'th", "/b/foo%3Fbar/+log/other%22na/me..ba/d%23name/we%20ird/pa%27th"),
+            breadcrumb("name",
+              "/b/foo%3Fbar/+log/other%22na/me..ba/d%23name/we%20ird/pa%27th/name")),
+        view.getBreadcrumbs());
+  }
+
+  private static ImmutableMap<String, String> breadcrumb(String text, String url) {
+    return ImmutableMap.of("text", text, "url", url);
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/LinkifierTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/LinkifierTest.java
new file mode 100644
index 0000000..6f6caf2
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/LinkifierTest.java
@@ -0,0 +1,113 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableMap;
+
+import junit.framework.TestCase;
+
+import javax.servlet.http.HttpServletRequest;
+
+/** Tests for {@link Linkifier}. */
+public class LinkifierTest extends TestCase {
+  private static final HttpServletRequest REQ = FakeHttpServletRequest.newRequest();
+
+  @Override
+  protected void setUp() throws Exception {
+  }
+
+  public void testlinkifyMessageNoMatch() throws Exception {
+    Linkifier l = new Linkifier(TestGitilesUrls.URLS);
+    assertEquals(ImmutableList.of(ImmutableMap.of("text", "some message text")),
+        l.linkify(FakeHttpServletRequest.newRequest(), "some message text"));
+  }
+
+  public void testlinkifyMessageUrl() throws Exception {
+    Linkifier l = new Linkifier(TestGitilesUrls.URLS);
+    assertEquals(ImmutableList.of(
+        ImmutableMap.of("text", "http://my/url", "url", "http://my/url")),
+        l.linkify(REQ, "http://my/url"));
+    assertEquals(ImmutableList.of(
+        ImmutableMap.of("text", "https://my/url", "url", "https://my/url")),
+        l.linkify(REQ, "https://my/url"));
+    assertEquals(ImmutableList.of(
+        ImmutableMap.of("text", "foo "),
+        ImmutableMap.of("text", "http://my/url", "url", "http://my/url"),
+        ImmutableMap.of("text", " bar")),
+        l.linkify(REQ, "foo http://my/url bar"));
+    assertEquals(ImmutableList.of(
+        ImmutableMap.of("text", "foo "),
+        ImmutableMap.of("text", "http://my/url", "url", "http://my/url"),
+        ImmutableMap.of("text", " bar "),
+        ImmutableMap.of("text", "http://my/other/url", "url", "http://my/other/url"),
+        ImmutableMap.of("text", " baz")),
+        l.linkify(REQ, "foo http://my/url bar http://my/other/url baz"));
+  }
+
+  public void testlinkifyMessageChangeIdNoGerrit() throws Exception {
+    Linkifier l = new Linkifier(new GitilesUrls() {
+      @Override
+      public String getBaseGerritUrl(HttpServletRequest req) {
+        return null;
+      }
+
+      @Override
+      public String getHostName(HttpServletRequest req) {
+        throw new UnsupportedOperationException();
+      }
+
+      @Override
+      public String getBaseGitUrl(HttpServletRequest req) {
+        throw new UnsupportedOperationException();
+      }
+    });
+    assertEquals(ImmutableList.of(ImmutableMap.of("text", "I0123456789")),
+        l.linkify(REQ, "I0123456789"));
+    assertEquals(ImmutableList.of(ImmutableMap.of("text", "Change-Id: I0123456789")),
+        l.linkify(REQ, "Change-Id: I0123456789"));
+    assertEquals(ImmutableList.of(ImmutableMap.of("text", "Change-Id: I0123456789 does not exist")),
+        l.linkify(REQ, "Change-Id: I0123456789 does not exist"));
+  }
+
+  public void testlinkifyMessageChangeId() throws Exception {
+    Linkifier l = new Linkifier(TestGitilesUrls.URLS);
+    assertEquals(ImmutableList.of(
+        ImmutableMap.of("text", "I0123456789",
+          "url", "http://test-host-review/foo/#/q/I0123456789,n,z")),
+        l.linkify(REQ, "I0123456789"));
+    assertEquals(ImmutableList.of(
+        ImmutableMap.of("text", "Change-Id: "),
+        ImmutableMap.of("text", "I0123456789",
+          "url", "http://test-host-review/foo/#/q/I0123456789,n,z")),
+        l.linkify(REQ, "Change-Id: I0123456789"));
+    assertEquals(ImmutableList.of(
+        ImmutableMap.of("text", "Change-Id: "),
+        ImmutableMap.of("text", "I0123456789",
+          "url", "http://test-host-review/foo/#/q/I0123456789,n,z"),
+        ImmutableMap.of("text", " exists")),
+        l.linkify(REQ, "Change-Id: I0123456789 exists"));
+  }
+
+  public void testlinkifyMessageUrlAndChangeId() throws Exception {
+    Linkifier l = new Linkifier(TestGitilesUrls.URLS);
+    assertEquals(ImmutableList.of(
+        ImmutableMap.of("text", "http://my/url/I0123456789", "url", "http://my/url/I0123456789"),
+        ImmutableMap.of("text", " is not change "),
+        ImmutableMap.of("text", "I0123456789",
+          "url", "http://test-host-review/foo/#/q/I0123456789,n,z")),
+        l.linkify(REQ, "http://my/url/I0123456789 is not change I0123456789"));
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/PaginatorTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/PaginatorTest.java
new file mode 100644
index 0000000..09661f2
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/PaginatorTest.java
@@ -0,0 +1,155 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkArgument;
+
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.Lists;
+
+import junit.framework.TestCase;
+
+import org.eclipse.jgit.junit.TestRepository;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevWalk;
+import org.eclipse.jgit.storage.dfs.DfsRepository;
+import org.eclipse.jgit.storage.dfs.DfsRepositoryDescription;
+import org.eclipse.jgit.storage.dfs.InMemoryRepository;
+
+import java.util.List;
+
+/** Unit tests for {@link LogServlet}. */
+public class PaginatorTest extends TestCase {
+  private TestRepository<DfsRepository> repo;
+  private RevWalk walk;
+
+  @Override
+  protected void setUp() throws Exception {
+    repo = new TestRepository<DfsRepository>(
+        new InMemoryRepository(new DfsRepositoryDescription("test")));
+    walk = new RevWalk(repo.getRepository());
+  }
+
+  @Override
+  protected void tearDown() throws Exception {
+    walk.release();
+  }
+
+  public void testStart() throws Exception {
+    List<RevCommit> commits = linearCommits(10);
+    walk.markStart(commits.get(9));
+    Paginator p = new Paginator(walk, 3, commits.get(9));
+    assertEquals(
+        ImmutableList.of(
+            commits.get(9),
+            commits.get(8),
+            commits.get(7)),
+        ImmutableList.copyOf(p));
+    assertNull(p.getPreviousStart());
+    assertEquals(commits.get(6), p.getNextStart());
+  }
+
+  public void testNoStartCommit() throws Exception {
+    List<RevCommit> commits = linearCommits(10);
+    walk.markStart(commits.get(9));
+    Paginator p = new Paginator(walk, 3, null);
+    assertEquals(
+        ImmutableList.of(
+            commits.get(9),
+            commits.get(8),
+            commits.get(7)),
+        ImmutableList.copyOf(p));
+    assertNull(p.getPreviousStart());
+    assertEquals(commits.get(6), p.getNextStart());
+  }
+
+  public void testLessThanOnePageIn() throws Exception {
+    List<RevCommit> commits = linearCommits(10);
+    walk.markStart(commits.get(9));
+    Paginator p = new Paginator(walk, 3, commits.get(8));
+    assertEquals(
+        ImmutableList.of(
+            commits.get(8),
+            commits.get(7),
+            commits.get(6)),
+        ImmutableList.copyOf(p));
+    assertEquals(commits.get(9), p.getPreviousStart());
+    assertEquals(commits.get(5), p.getNextStart());
+  }
+
+  public void testAtLeastOnePageIn() throws Exception {
+    List<RevCommit> commits = linearCommits(10);
+    walk.markStart(commits.get(9));
+    Paginator p = new Paginator(walk, 3, commits.get(7));
+    assertEquals(
+        ImmutableList.of(
+            commits.get(7),
+            commits.get(6),
+            commits.get(5)),
+        ImmutableList.copyOf(p));
+    assertEquals(commits.get(9), p.getPreviousStart());
+    assertEquals(commits.get(4), p.getNextStart());
+  }
+
+  public void testEnd() throws Exception {
+    List<RevCommit> commits = linearCommits(10);
+    walk.markStart(commits.get(9));
+    Paginator p = new Paginator(walk, 3, commits.get(2));
+    assertEquals(
+        ImmutableList.of(
+            commits.get(2),
+            commits.get(1),
+            commits.get(0)),
+        ImmutableList.copyOf(p));
+    assertEquals(commits.get(5), p.getPreviousStart());
+    assertNull(p.getNextStart());
+  }
+
+  public void testOnePastEnd() throws Exception {
+    List<RevCommit> commits = linearCommits(10);
+    walk.markStart(commits.get(9));
+    Paginator p = new Paginator(walk, 3, commits.get(1));
+    assertEquals(
+        ImmutableList.of(
+            commits.get(1),
+            commits.get(0)),
+        ImmutableList.copyOf(p));
+    assertEquals(commits.get(4), p.getPreviousStart());
+    assertNull(p.getNextStart());
+  }
+
+  public void testManyPastEnd() throws Exception {
+    List<RevCommit> commits = linearCommits(10);
+    walk.markStart(commits.get(9));
+    Paginator p = new Paginator(walk, 5, commits.get(1));
+    assertEquals(
+        ImmutableList.of(
+            commits.get(1),
+            commits.get(0)),
+        ImmutableList.copyOf(p));
+    assertEquals(commits.get(6), p.getPreviousStart());
+    assertNull(p.getNextStart());
+  }
+
+  private List<RevCommit> linearCommits(int n) throws Exception {
+    checkArgument(n > 0);
+    List<RevCommit> commits = Lists.newArrayList();
+    commits.add(repo.commit().create());
+    for (int i = 1; i < 10; i++) {
+      commits.add(repo.commit().parent(commits.get(commits.size() - 1)).create());
+    }
+    return commits;
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/RepositoryIndexServletTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/RepositoryIndexServletTest.java
new file mode 100644
index 0000000..3071f70
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/RepositoryIndexServletTest.java
@@ -0,0 +1,132 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.gitiles.TestGitilesUrls.URLS;
+
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableMap;
+
+import junit.framework.TestCase;
+
+import org.eclipse.jgit.junit.TestRepository;
+import org.eclipse.jgit.storage.dfs.DfsRepository;
+import org.eclipse.jgit.storage.dfs.DfsRepositoryDescription;
+import org.eclipse.jgit.storage.dfs.InMemoryRepository;
+
+import java.io.IOException;
+import java.util.Map;
+
+import javax.servlet.http.HttpServletRequest;
+
+/** Tests for {@link RepositoryIndexServlet}. */
+public class RepositoryIndexServletTest extends TestCase {
+  private TestRepository<DfsRepository> repo;
+  private RepositoryIndexServlet servlet;
+
+  @Override
+  protected void setUp() throws Exception {
+    repo = new TestRepository<DfsRepository>(
+        new InMemoryRepository(new DfsRepositoryDescription("test")));
+    servlet = new RepositoryIndexServlet(
+        new DefaultRenderer(),
+        new TestGitilesAccess(repo.getRepository()));
+  }
+
+  public void testEmpty() throws Exception {
+    Map<String, ?> data = buildData();
+    assertEquals(ImmutableList.of(), data.get("branches"));
+    assertEquals(ImmutableList.of(), data.get("tags"));
+  }
+
+  public void testBranchesAndTags() throws Exception {
+    repo.branch("refs/heads/foo").commit().create();
+    repo.branch("refs/heads/bar").commit().create();
+    repo.branch("refs/tags/baz").commit().create();
+    repo.branch("refs/nope/quux").commit().create();
+    Map<String, ?> data = buildData();
+
+    assertEquals(
+        ImmutableList.of(
+            ref("/b/test/+/bar", "bar"),
+            ref("/b/test/+/foo", "foo")),
+        data.get("branches"));
+    assertEquals(
+        ImmutableList.of(
+            ref("/b/test/+/baz", "baz")),
+        data.get("tags"));
+  }
+
+  public void testAmbiguousBranch() throws Exception {
+    repo.branch("refs/heads/foo").commit().create();
+    repo.branch("refs/heads/bar").commit().create();
+    repo.branch("refs/tags/foo").commit().create();
+    Map<String, ?> data = buildData();
+
+    assertEquals(
+        ImmutableList.of(
+            ref("/b/test/+/bar", "bar"),
+            ref("/b/test/+/refs/heads/foo", "foo")),
+        data.get("branches"));
+    assertEquals(
+        ImmutableList.of(
+            // refs/tags/ is searched before refs/heads/, so this does not
+            // appear ambiguous.
+            ref("/b/test/+/foo", "foo")),
+        data.get("tags"));
+  }
+
+  public void testAmbiguousRelativeToNonBranchOrTag() throws Exception {
+    repo.branch("refs/foo").commit().create();
+    repo.branch("refs/heads/foo").commit().create();
+    repo.branch("refs/tags/foo").commit().create();
+    Map<String, ?> data = buildData();
+
+    assertEquals(
+        ImmutableList.of(
+            ref("/b/test/+/refs/heads/foo", "foo")),
+        data.get("branches"));
+    assertEquals(
+        ImmutableList.of(
+            ref("/b/test/+/refs/tags/foo", "foo")),
+        data.get("tags"));
+  }
+
+  public void testRefsHeads() throws Exception {
+    repo.branch("refs/heads/foo").commit().create();
+    repo.branch("refs/heads/refs/heads/foo").commit().create();
+    Map<String, ?> data = buildData();
+
+    assertEquals(
+        ImmutableList.of(
+            ref("/b/test/+/foo", "foo"),
+            ref("/b/test/+/refs/heads/refs/heads/foo", "refs/heads/foo")),
+        data.get("branches"));
+  }
+
+  private Map<String, ?> buildData() throws IOException {
+    HttpServletRequest req = FakeHttpServletRequest.newRequest(repo.getRepository());
+    ViewFilter.setView(req, GitilesView.repositoryIndex()
+        .setHostName(URLS.getHostName(req))
+        .setServletPath(req.getServletPath())
+        .setRepositoryName("test")
+        .build());
+    return servlet.buildData(req);
+  }
+
+  private Map<String, String> ref(String url, String name) {
+    return ImmutableMap.of("url", url, "name", name);
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/RevisionParserTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/RevisionParserTest.java
new file mode 100644
index 0000000..23033e0
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/RevisionParserTest.java
@@ -0,0 +1,273 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static org.eclipse.jgit.lib.Constants.OBJ_BLOB;
+import static org.eclipse.jgit.lib.Constants.OBJ_COMMIT;
+import static org.eclipse.jgit.lib.Constants.OBJ_TAG;
+
+import com.google.common.cache.CacheBuilder;
+import com.google.gitiles.RevisionParser.Result;
+
+import junit.framework.TestCase;
+
+import org.eclipse.jgit.junit.TestRepository;
+import org.eclipse.jgit.revwalk.RevBlob;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.revwalk.RevTag;
+import org.eclipse.jgit.storage.dfs.DfsRepository;
+import org.eclipse.jgit.storage.dfs.DfsRepositoryDescription;
+import org.eclipse.jgit.storage.dfs.InMemoryRepository;
+
+/** Tests for the revision parser. */
+public class RevisionParserTest extends TestCase {
+  private TestRepository<DfsRepository> repo;
+  private RevisionParser parser;
+
+  @Override
+  protected void setUp() throws Exception {
+    repo = new TestRepository<DfsRepository>(
+        new InMemoryRepository(new DfsRepositoryDescription("test")));
+    parser = new RevisionParser(
+        repo.getRepository(),
+        new TestGitilesAccess(repo.getRepository()).forRequest(null),
+        new VisibilityCache(false, CacheBuilder.newBuilder().maximumSize(0)));
+  }
+
+  public void testParseRef() throws Exception {
+    RevCommit master = repo.branch("refs/heads/master").commit().create();
+    assertEquals(new Result(Revision.peeled("master", master)),
+        parser.parse("master"));
+    assertEquals(new Result(Revision.peeled("refs/heads/master", master)),
+        parser.parse("refs/heads/master"));
+    assertNull(parser.parse("refs//heads//master"));
+  }
+
+  public void testParseRefParentExpression() throws Exception {
+    RevCommit root = repo.commit().create();
+    RevCommit parent1 = repo.commit().parent(root).create();
+    RevCommit parent2 = repo.commit().parent(root).create();
+    RevCommit merge = repo.branch("master").commit()
+        .parent(parent1)
+        .parent(parent2)
+        .create();
+    assertEquals(new Result(Revision.peeled("master", merge)), parser.parse("master"));
+    assertEquals(new Result(Revision.peeled("master^", parent1)), parser.parse("master^"));
+    assertEquals(new Result(Revision.peeled("master~1", parent1)), parser.parse("master~1"));
+    assertEquals(new Result(Revision.peeled("master^2", parent2)), parser.parse("master^2"));
+    assertEquals(new Result(Revision.peeled("master~2", root)), parser.parse("master~2"));
+  }
+
+  public void testParseCommitShaVisibleFromHead() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit commit = repo.branch("master").commit().parent(parent).create();
+    assertEquals(new Result(Revision.peeled(commit.name(), commit)), parser.parse(commit.name()));
+    assertEquals(new Result(Revision.peeled(parent.name(), parent)), parser.parse(parent.name()));
+
+    String abbrev = commit.name().substring(0, 6);
+    assertEquals(new Result(Revision.peeled(abbrev, commit)), parser.parse(abbrev));
+  }
+
+  public void testParseCommitShaVisibleFromTag() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit commit = repo.commit().parent(parent).create();
+    repo.branch("master").commit().create();
+    repo.update("refs/tags/tag", repo.tag("tag", commit));
+
+    assertEquals(new Result(Revision.peeled(commit.name(), commit)), parser.parse(commit.name()));
+    assertEquals(new Result(Revision.peeled(parent.name(), parent)), parser.parse(parent.name()));
+  }
+
+  public void testParseCommitShaVisibleFromOther() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit commit = repo.commit().parent(parent).create();
+    repo.branch("master").commit().create();
+    repo.update("refs/tags/tag", repo.tag("tag", repo.commit().create()));
+    repo.update("refs/meta/config", commit);
+
+    assertEquals(new Result(Revision.peeled(commit.name(), commit)), parser.parse(commit.name()));
+    assertEquals(new Result(Revision.peeled(parent.name(), parent)), parser.parse(parent.name()));
+  }
+
+  public void testParseCommitShaVisibleFromChange() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit commit = repo.commit().parent(parent).create();
+    repo.branch("master").commit().create();
+    repo.update("refs/changes/01/0001", commit);
+
+    // Matches exactly.
+    assertEquals(new Result(Revision.peeled(commit.name(), commit)), parser.parse(commit.name()));
+    // refs/changes/* is excluded from ancestry search.
+    assertEquals(null, parser.parse(parent.name()));
+  }
+
+  public void testParseNonVisibleCommitSha() throws Exception {
+    RevCommit other = repo.commit().create();
+    RevCommit master = repo.branch("master").commit().create();
+    assertEquals(null, parser.parse(other.name()));
+
+    repo.branch("other").update(other);
+    assertEquals(new Result(Revision.peeled(other.name(), other)), parser.parse(other.name()));
+  }
+
+  public void testParseDiffRevisions() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit commit = repo.branch("master").commit().parent(parent).create();
+    RevCommit other = repo.branch("other").commit().create();
+
+    assertEquals(
+        new Result(
+            Revision.peeled("master", commit),
+            Revision.peeled("master^", parent),
+            15),
+        parser.parse("master^..master"));
+    assertEquals(
+        new Result(
+            Revision.peeled("master", commit),
+            Revision.peeled("master^", parent),
+            15),
+        parser.parse("master^..master/"));
+    assertEquals(
+        new Result(
+            Revision.peeled("master", commit),
+            Revision.peeled("master^", parent),
+            15),
+        parser.parse("master^..master/path/to/a/file"));
+    assertEquals(
+        new Result(
+            Revision.peeled("master", commit),
+            Revision.peeled("master^", parent),
+            15),
+        parser.parse("master^..master/path/to/a/..file"));
+    assertEquals(
+        new Result(
+            Revision.peeled("refs/heads/master", commit),
+            Revision.peeled("refs/heads/master^", parent),
+            37),
+      parser.parse("refs/heads/master^..refs/heads/master"));
+    assertEquals(
+        new Result(
+            Revision.peeled("master", commit),
+            Revision.peeled("master~1", parent),
+            16),
+        parser.parse("master~1..master"));
+    // TODO(dborowitz): 2a2362fbb in JGit causes master~2 to resolve to master
+    // rather than null. Uncomment when upstream regression is fixed.
+    //assertNull(parser.parse("master~2..master"));
+    assertEquals(
+        new Result(
+            Revision.peeled("master", commit),
+            Revision.peeled("other", other),
+            13),
+        parser.parse("other..master"));
+  }
+
+  public void testParseFirstParentExpression() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit commit = repo.branch("master").commit().parent(parent).create();
+
+    assertEquals(
+        new Result(
+            Revision.peeled("master", commit),
+            Revision.peeled("master^", parent),
+            8),
+        parser.parse("master^!"));
+    assertEquals(
+        new Result(
+            Revision.peeled("master^", parent),
+            Revision.NULL,
+            9),
+        parser.parse("master^^!"));
+    assertEquals(
+        new Result(
+            Revision.peeled(parent.name(), parent),
+            Revision.NULL,
+            42),
+        parser.parse(parent.name() + "^!"));
+
+    RevTag tag = repo.update("refs/tags/tag", repo.tag("tag", commit));
+    assertEquals(
+        new Result(
+            Revision.peeled("tag", commit),
+            Revision.peeled("tag^", parent),
+            5),
+        parser.parse("tag^!"));
+    assertEquals(
+        new Result(
+            Revision.peeled("tag^", parent),
+            Revision.NULL,
+            6),
+        parser.parse("tag^^!"));
+  }
+
+  public void testNonVisibleDiffShas() throws Exception {
+    RevCommit other = repo.commit().create();
+    RevCommit master = repo.branch("master").commit().create();
+    assertEquals(null, parser.parse("other..master"));
+    assertEquals(null, parser.parse("master..other"));
+
+    repo.branch("other").update(other);
+    assertEquals(
+        new Result(
+            Revision.peeled("master", master),
+            Revision.peeled("other", other),
+            13),
+        parser.parse("other..master"));
+    assertEquals(
+        new Result(
+            Revision.peeled("other", other),
+            Revision.peeled("master", master),
+            13),
+        parser.parse("master..other"));
+  }
+
+  public void testParseTag() throws Exception {
+    RevCommit master = repo.branch("master").commit().create();
+    RevTag masterTag = repo.update("refs/tags/master-tag", repo.tag("master-tag", master));
+    RevTag masterTagTag = repo.update("refs/tags/master-tag-tag",
+        repo.tag("master-tag-tag", master));
+
+    assertEquals(new Result(
+            new Revision("master-tag", masterTag, OBJ_TAG, master, OBJ_COMMIT)),
+        parser.parse("master-tag"));
+    assertEquals(new Result(
+            new Revision("master-tag-tag", masterTagTag, OBJ_TAG, master, OBJ_COMMIT)),
+        parser.parse("master-tag-tag"));
+
+    RevBlob blob = repo.update("refs/tags/blob", repo.blob("blob"));
+    RevTag blobTag = repo.update("refs/tags/blob-tag", repo.tag("blob-tag", blob));
+    assertEquals(new Result(Revision.peeled("blob", blob)), parser.parse("blob"));
+    assertEquals(new Result(new Revision("blob-tag", blobTag, OBJ_TAG, blob, OBJ_BLOB)),
+        parser.parse("blob-tag"));
+  }
+
+  public void testParseUnsupportedRevisionExpressions() throws Exception {
+    RevBlob blob = repo.blob("blob contents");
+    RevCommit master = repo.branch("master").commit().add("blob", blob).create();
+
+    assertEquals(master, repo.getRepository().resolve("master^{}"));
+    assertEquals(null, parser.parse("master^{}"));
+
+    assertEquals(master, repo.getRepository().resolve("master^{commit}"));
+    assertEquals(null, parser.parse("master^{commit}"));
+
+    assertEquals(blob, repo.getRepository().resolve("master:blob"));
+    assertEquals(null, parser.parse("master:blob"));
+
+    // TestRepository has no simple way of setting the reflog.
+    //assertEquals(null, repo.getRepository().resolve("master@{0}"));
+    assertEquals(null, parser.parse("master@{0}"));
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/TestGitilesAccess.java b/gitiles-servlet/src/test/java/com/google/gitiles/TestGitilesAccess.java
new file mode 100644
index 0000000..0b31e24
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/TestGitilesAccess.java
@@ -0,0 +1,64 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.common.base.Preconditions.checkNotNull;
+
+import org.eclipse.jgit.storage.dfs.DfsRepository;
+
+import java.util.Map;
+import java.util.Set;
+
+import javax.servlet.http.HttpServletRequest;
+
+/** Gitiles access for testing. */
+public class TestGitilesAccess implements GitilesAccess.Factory {
+  private final DfsRepository repo;
+
+  public TestGitilesAccess(DfsRepository repo) {
+    this.repo = checkNotNull(repo);
+  }
+
+  @Override
+  public GitilesAccess forRequest(final HttpServletRequest req) {
+    return new GitilesAccess() {
+      @Override
+      public Map<String, RepositoryDescription> listRepositories(Set<String> branches) {
+        // TODO(dborowitz): Implement this, using the DfsRepositoryDescriptions to
+        // get the repository names.
+        throw new UnsupportedOperationException();
+      }
+
+      @Override
+      public Object getUserKey() {
+        return "a user";
+      }
+
+      @Override
+      public String getRepositoryName() {
+        return repo.getDescription().getRepositoryName();
+      }
+
+      @Override
+      public RepositoryDescription getRepositoryDescription() {
+        RepositoryDescription d = new RepositoryDescription();
+        d.name = getRepositoryName();
+        d.description = "a test data set";
+        d.cloneUrl = TestGitilesUrls.URLS.getBaseGitUrl(req) + "/" + d.name;
+        return d;
+      }
+    };
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/TestGitilesUrls.java b/gitiles-servlet/src/test/java/com/google/gitiles/TestGitilesUrls.java
new file mode 100644
index 0000000..f8a0883
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/TestGitilesUrls.java
@@ -0,0 +1,40 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import javax.servlet.http.HttpServletRequest;
+
+/** {@link GitilesUrls} for testing. */
+public class TestGitilesUrls implements GitilesUrls {
+  public static final GitilesUrls URLS = new TestGitilesUrls();
+
+  @Override
+  public String getHostName(HttpServletRequest req) {
+    return "test-host";
+  }
+
+  @Override
+  public String getBaseGitUrl(HttpServletRequest req) {
+    return "git://test-host/foo";
+  }
+
+  @Override
+  public String getBaseGerritUrl(HttpServletRequest req) {
+    return "http://test-host-review/foo/";
+  }
+
+  private TestGitilesUrls() {
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/TreeSoyDataTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/TreeSoyDataTest.java
new file mode 100644
index 0000000..85423d9
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/TreeSoyDataTest.java
@@ -0,0 +1,61 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.gitiles.TreeSoyData.getTargetDisplayName;
+import static com.google.gitiles.TreeSoyData.resolveTargetUrl;
+
+import com.google.common.base.Strings;
+
+import junit.framework.TestCase;
+
+import org.eclipse.jgit.lib.ObjectId;
+
+/** Tests for {@link TreeSoyData}. */
+public class TreeSoyDataTest extends TestCase {
+  public void testGetTargetDisplayName() throws Exception {
+    assertEquals("foo", getTargetDisplayName("foo"));
+    assertEquals("foo/bar", getTargetDisplayName("foo/bar"));
+    assertEquals("a/a/a/a/a/a/a/a/a/a/bar",
+        getTargetDisplayName(Strings.repeat("a/", 10) + "bar"));
+    assertEquals("a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/a/bar",
+        getTargetDisplayName(Strings.repeat("a/", 34) + "bar"));
+    assertEquals(".../bar", getTargetDisplayName(Strings.repeat("a/", 35) + "bar"));
+    assertEquals(".../bar", getTargetDisplayName(Strings.repeat("a/", 100) + "bar"));
+    assertEquals("aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa",
+        getTargetDisplayName(Strings.repeat("a", 80)));
+  }
+
+  public void testResolveTargetUrl() throws Exception {
+    ObjectId id = ObjectId.fromString("abcd1234abcd1234abcd1234abcd1234abcd1234");
+    GitilesView view = GitilesView.path()
+        .setServletPath("/x")
+        .setHostName("host")
+        .setRepositoryName("repo")
+        .setRevision(Revision.unpeeled("m", id))
+        .setTreePath("a/b/c")
+        .build();
+    assertNull(resolveTargetUrl(view, "/foo"));
+    assertEquals("/x/repo/+/m/a", resolveTargetUrl(view, "../../"));
+    assertEquals("/x/repo/+/m/a", resolveTargetUrl(view, ".././../"));
+    assertEquals("/x/repo/+/m/a", resolveTargetUrl(view, "..//../"));
+    assertEquals("/x/repo/+/m/a/d", resolveTargetUrl(view, "../../d"));
+    assertEquals("/x/repo/+/m/", resolveTargetUrl(view, "../../.."));
+    assertEquals("/x/repo/+/m/a/d/e", resolveTargetUrl(view, "../../d/e"));
+    assertEquals("/x/repo/+/m/a/b", resolveTargetUrl(view, "../d/../e/../"));
+    assertNull(resolveTargetUrl(view, "../../../../"));
+    assertNull(resolveTargetUrl(view, "../../a/../../.."));
+  }
+}
diff --git a/gitiles-servlet/src/test/java/com/google/gitiles/ViewFilterTest.java b/gitiles-servlet/src/test/java/com/google/gitiles/ViewFilterTest.java
new file mode 100644
index 0000000..1427b9e
--- /dev/null
+++ b/gitiles-servlet/src/test/java/com/google/gitiles/ViewFilterTest.java
@@ -0,0 +1,346 @@
+// Copyright 2012 Google Inc. All Rights Reserved.
+//
+// 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.google.gitiles;
+
+import static com.google.gitiles.FakeHttpServletRequest.newRequest;
+import static com.google.gitiles.GitilesFilter.REPO_PATH_REGEX;
+import static com.google.gitiles.GitilesFilter.REPO_REGEX;
+import static com.google.gitiles.GitilesFilter.ROOT_REGEX;
+
+import com.google.common.collect.ImmutableList;
+import com.google.common.util.concurrent.Atomics;
+import com.google.gitiles.GitilesView.Type;
+
+import junit.framework.TestCase;
+
+import org.eclipse.jgit.http.server.glue.MetaFilter;
+import org.eclipse.jgit.http.server.glue.MetaServlet;
+import org.eclipse.jgit.junit.TestRepository;
+import org.eclipse.jgit.revwalk.RevCommit;
+import org.eclipse.jgit.storage.dfs.DfsRepository;
+import org.eclipse.jgit.storage.dfs.DfsRepositoryDescription;
+import org.eclipse.jgit.storage.dfs.InMemoryRepository;
+
+import java.io.IOException;
+import java.util.concurrent.atomic.AtomicReference;
+import java.util.regex.Pattern;
+
+import javax.servlet.ServletException;
+import javax.servlet.http.HttpServlet;
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+/** Tests for the view filter. */
+public class ViewFilterTest extends TestCase {
+  private TestRepository<DfsRepository> repo;
+
+  @Override
+  protected void setUp() throws Exception {
+    repo = new TestRepository<DfsRepository>(
+        new InMemoryRepository(new DfsRepositoryDescription("test")));
+  }
+
+  public void testNoCommand() throws Exception {
+    assertEquals(Type.HOST_INDEX, getView("/").getType());
+    assertEquals(Type.REPOSITORY_INDEX, getView("/repo").getType());
+    assertNull(getView("/repo/+"));
+    assertNull(getView("/repo/+/"));
+  }
+
+  public void testAutoCommand() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit master = repo.branch("refs/heads/master").commit().parent(parent).create();
+    String hex = master.name();
+    String hexBranch = hex.substring(0, 10);
+    RevCommit hexCommit = repo.branch(hexBranch).commit().create();
+
+    assertEquals(Type.LOG, getView("/repo/+/master").getType());
+    assertEquals(Type.LOG, getView("/repo/+/" + hexBranch).getType());
+    assertEquals(Type.REVISION, getView("/repo/+/" + hex).getType());
+    assertEquals(Type.REVISION, getView("/repo/+/" + hex.substring(0, 7)).getType());
+    assertEquals(Type.PATH, getView("/repo/+/master/").getType());
+    assertEquals(Type.PATH, getView("/repo/+/" + hex + "/").getType());
+    assertEquals(Type.DIFF, getView("/repo/+/master^..master").getType());
+    assertEquals(Type.DIFF, getView("/repo/+/master^..master/").getType());
+    assertEquals(Type.DIFF, getView("/repo/+/" + parent.name() + ".." + hex + "/").getType());
+  }
+
+  public void testHostIndex() throws Exception {
+    GitilesView view = getView("/");
+    assertEquals(Type.HOST_INDEX, view.getType());
+    assertEquals("test-host", view.getHostName());
+    assertNull(view.getRepositoryName());
+    assertEquals(Revision.NULL, view.getRevision());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertNull(view.getTreePath());
+  }
+
+  public void testRepositoryIndex() throws Exception {
+    GitilesView view = getView("/repo");
+    assertEquals(Type.REPOSITORY_INDEX, view.getType());
+    assertEquals("repo", view.getRepositoryName());
+    assertEquals(Revision.NULL, view.getRevision());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertNull(view.getTreePath());
+  }
+
+  public void testBranches() throws Exception {
+    RevCommit master = repo.branch("refs/heads/master").commit().create();
+    RevCommit stable = repo.branch("refs/heads/stable").commit().create();
+    GitilesView view;
+
+    view = getView("/repo/+show/master");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertNull(view.getTreePath());
+
+    view = getView("/repo/+show/heads/master");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("heads/master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertNull(view.getTreePath());
+
+    view = getView("/repo/+show/refs/heads/master");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("refs/heads/master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertNull(view.getTreePath());
+
+    view = getView("/repo/+show/stable");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("stable", view.getRevision().getName());
+    assertEquals(stable, view.getRevision().getId());
+    assertNull(view.getTreePath());
+  }
+
+  public void testAmbiguousBranchAndTag() throws Exception {
+    RevCommit branch = repo.branch("refs/heads/name").commit().create();
+    RevCommit tag = repo.branch("refs/tags/name").commit().create();
+    GitilesView view;
+
+    view = getView("/repo/+show/name");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("name", view.getRevision().getName());
+    assertEquals(tag, view.getRevision().getId());
+    assertNull(view.getTreePath());
+
+    view = getView("/repo/+show/heads/name");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("heads/name", view.getRevision().getName());
+    assertEquals(branch, view.getRevision().getId());
+    assertNull(view.getTreePath());
+
+    view = getView("/repo/+show/refs/heads/name");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("refs/heads/name", view.getRevision().getName());
+    assertEquals(branch, view.getRevision().getId());
+    assertNull(view.getTreePath());
+
+    view = getView("/repo/+show/tags/name");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("tags/name", view.getRevision().getName());
+    assertEquals(tag, view.getRevision().getId());
+    assertNull(view.getTreePath());
+
+    view = getView("/repo/+show/refs/tags/name");
+    assertEquals(Type.REVISION, view.getType());
+    assertEquals("refs/tags/name", view.getRevision().getName());
+    assertEquals(tag, view.getRevision().getId());
+    assertNull(view.getTreePath());
+  }
+
+  public void testPath() throws Exception {
+    RevCommit master = repo.branch("refs/heads/master").commit().create();
+    GitilesView view;
+
+    view = getView("/repo/+show/master/");
+    assertEquals(Type.PATH, view.getType());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("", view.getTreePath());
+
+    view = getView("/repo/+show/master/foo");
+    assertEquals(Type.PATH, view.getType());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("foo", view.getTreePath());
+
+    view = getView("/repo/+show/master/foo/");
+    assertEquals(Type.PATH, view.getType());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("foo", view.getTreePath());
+
+    view = getView("/repo/+show/master/foo/bar");
+    assertEquals(Type.PATH, view.getType());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("foo/bar", view.getTreePath());
+  }
+
+  public void testMultipleSlashes() throws Exception {
+    RevCommit master = repo.branch("refs/heads/master").commit().create();
+    GitilesView view;
+
+    assertEquals(Type.HOST_INDEX, getView("//").getType());
+    assertEquals(Type.REPOSITORY_INDEX, getView("//repo").getType());
+    assertEquals(Type.REPOSITORY_INDEX, getView("//repo//").getType());
+    assertNull(getView("/repo/+//master"));
+    assertNull(getView("/repo/+/refs//heads//master"));
+    assertNull(getView("/repo/+//master//"));
+    assertNull(getView("/repo/+//master/foo//bar"));
+  }
+
+  public void testDiff() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit master = repo.branch("refs/heads/master").commit().parent(parent).create();
+    GitilesView view;
+
+    view = getView("/repo/+diff/master^..master");
+    assertEquals(Type.DIFF, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("master^", view.getOldRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("", view.getTreePath());
+
+    view = getView("/repo/+diff/master^..master/");
+    assertEquals(Type.DIFF, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("master^", view.getOldRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("", view.getTreePath());
+
+    view = getView("/repo/+diff/master^..master/foo");
+    assertEquals(Type.DIFF, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("master^", view.getOldRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("foo", view.getTreePath());
+
+    view = getView("/repo/+diff/refs/heads/master^..refs/heads/master");
+    assertEquals(Type.DIFF, view.getType());
+    assertEquals("refs/heads/master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("refs/heads/master^", view.getOldRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("", view.getTreePath());
+  }
+
+  public void testDiffAgainstEmptyCommit() throws Exception {
+    RevCommit master = repo.branch("refs/heads/master").commit().create();
+    GitilesView view = getView("/repo/+diff/master^!");
+    assertEquals(Type.DIFF, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertEquals("", view.getTreePath());
+  }
+
+  public void testLog() throws Exception {
+    RevCommit parent = repo.commit().create();
+    RevCommit master = repo.branch("refs/heads/master").commit().parent(parent).create();
+    GitilesView view;
+
+    assertNull(getView("/repo/+log"));
+    assertNull(getView("/repo/+log/"));
+
+    view = getView("/repo/+log/master");
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertEquals("", view.getTreePath());
+
+    view = getView("/repo/+log/master/");
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertEquals("", view.getTreePath());
+
+    view = getView("/repo/+log/master/foo");
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals(Revision.NULL, view.getOldRevision());
+    assertEquals("foo", view.getTreePath());
+
+    view = getView("/repo/+log/master^..master");
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("master^", view.getOldRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("", view.getTreePath());
+
+    view = getView("/repo/+log/master^..master/");
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("master^", view.getOldRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("", view.getTreePath());
+
+    view = getView("/repo/+log/master^..master/foo");
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("master^", view.getOldRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("foo", view.getTreePath());
+
+    view = getView("/repo/+log/refs/heads/master^..refs/heads/master");
+    assertEquals(Type.LOG, view.getType());
+    assertEquals("refs/heads/master", view.getRevision().getName());
+    assertEquals(master, view.getRevision().getId());
+    assertEquals("refs/heads/master^", view.getOldRevision().getName());
+    assertEquals(parent, view.getOldRevision().getId());
+    assertEquals("", view.getTreePath());
+  }
+
+  private GitilesView getView(String pathAndQuery) throws ServletException, IOException {
+    final AtomicReference<GitilesView> view = Atomics.newReference();
+    HttpServlet testServlet = new HttpServlet() {
+      @Override
+      protected void doGet(HttpServletRequest req, HttpServletResponse res) {
+        view.set(ViewFilter.getView(req));
+      }
+    };
+
+    ViewFilter vf = new ViewFilter(
+        new TestGitilesAccess(repo.getRepository()),
+        TestGitilesUrls.URLS,
+        new VisibilityCache(false));
+    MetaFilter mf = new MetaFilter();
+
+    for (Pattern p : ImmutableList.of(ROOT_REGEX, REPO_REGEX, REPO_PATH_REGEX)) {
+      mf.serveRegex(p)
+          .through(vf)
+          .with(testServlet);
+    }
+
+    FakeHttpServletRequest req = newRequest(repo.getRepository());
+    int q = pathAndQuery.indexOf('?');
+    if (q > 0) {
+      req.setPathInfo(pathAndQuery.substring(0, q));
+      req.setQueryString(pathAndQuery.substring(q + 1));
+    } else {
+      req.setPathInfo(pathAndQuery);
+    }
+    new MetaServlet(mf){}.service(req, new FakeHttpServletResponse());
+
+    return view.get();
+  }
+}
diff --git a/gitiles-war/pom.xml b/gitiles-war/pom.xml
new file mode 100644
index 0000000..9ad21f1
--- /dev/null
+++ b/gitiles-war/pom.xml
@@ -0,0 +1,73 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!--
+Copyright 2012 Google Inc. All Rights Reserved.
+
+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.
+-->
+<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
+  <modelVersion>4.0.0</modelVersion>
+
+  <parent>
+    <groupId>com.google.gitiles</groupId>
+    <artifactId>gitiles-parent</artifactId>
+    <version>1.0-SNAPSHOT</version>
+  </parent>
+
+  <artifactId>gitiles-war</artifactId>
+  <packaging>war</packaging>
+  <name>Gitiles - WAR</name>
+
+  <description>
+    Gitiles packaged as a standard web application archive
+  </description>
+
+  <dependencies>
+    <dependency>
+      <groupId>com.google.gitiles</groupId>
+      <artifactId>gitiles-servlet</artifactId>
+      <version>${project.version}</version>
+      <scope>runtime</scope>
+    </dependency>
+
+    <dependency>
+      <groupId>org.apache.tomcat</groupId>
+      <artifactId>servlet-api</artifactId>
+      <scope>provided</scope>
+    </dependency>
+
+    <dependency>
+      <groupId>org.eclipse.jetty</groupId>
+      <artifactId>jetty-server</artifactId>
+      <version>${jettyVersion}</version>
+      <scope>provided</scope>
+    </dependency>
+  </dependencies>
+
+  <build>
+    <plugins>
+      <plugin>
+        <groupId>org.apache.maven.plugins</groupId>
+        <artifactId>maven-war-plugin</artifactId>
+        <configuration>
+          <webResources>
+            <resource>
+              <directory>../gitiles-servlet/src/main/resources/com/google/gitiles/static</directory>
+              <targetPath>+static</targetPath>
+            </resource>
+          </webResources>
+        </configuration>
+      </plugin>
+    </plugins>
+  </build>
+</project>
diff --git a/gitiles-war/src/main/webapp/WEB-INF/web.xml b/gitiles-war/src/main/webapp/WEB-INF/web.xml
new file mode 100644
index 0000000..b4ad57d
--- /dev/null
+++ b/gitiles-war/src/main/webapp/WEB-INF/web.xml
@@ -0,0 +1,35 @@
+<!DOCTYPE web-app PUBLIC
+ "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
+ "http://java.sun.com/dtd/web-app_2_3.dtd" >
+<!--
+  Copyright 2012 Google Inc. All Rights Reserved.
+
+  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.
+-->
+<web-app>
+  <display-name>Gitiles</display-name>
+
+  <servlet>
+    <servlet-name>gitiles</servlet-name>
+    <servlet-class>com.google.gitiles.GitilesServlet</servlet-class>
+    <init-param>
+      <param-name>configPath</param-name>
+      <param-value>gitiles.config</param-value>
+    </init-param>
+  </servlet>
+
+  <servlet-mapping>
+    <servlet-name>gitiles</servlet-name>
+    <url-pattern>/*</url-pattern>
+  </servlet-mapping>
+</web-app>
diff --git a/gitiles-war/webdefault.xml b/gitiles-war/webdefault.xml
new file mode 100644
index 0000000..8eaddc4
--- /dev/null
+++ b/gitiles-war/webdefault.xml
@@ -0,0 +1,28 @@
+<!DOCTYPE web-app PUBLIC
+ "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
+ "http://java.sun.com/dtd/web-app_2_3.dtd" >
+<!--
+  Copyright 2012 Google Inc. All Rights Reserved.
+
+  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.
+-->
+<web-app>
+  <servlet>
+    <servlet-name>default</servlet-name>
+    <servlet-class>org.eclipse.jetty.servlet.DefaultServlet</servlet-class>
+  </servlet>
+  <servlet-mapping>
+    <servlet-name>default</servlet-name>
+    <url-pattern>/+static/*</url-pattern>
+  </servlet-mapping>
+</web-app>
diff --git a/pom.xml b/pom.xml
new file mode 100644
index 0000000..aedba11
--- /dev/null
+++ b/pom.xml
@@ -0,0 +1,204 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!--
+Copyright 2012 Google Inc. All Rights Reserved.
+
+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.
+-->
+<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
+  <modelVersion>4.0.0</modelVersion>
+
+  <groupId>com.google.gitiles</groupId>
+  <artifactId>gitiles-parent</artifactId>
+  <packaging>pom</packaging>
+  <version>1.0-SNAPSHOT</version>
+
+  <name>Gitiles - Parent</name>
+  <url>https://gerrit.googlesource.com/gitiles</url>
+
+  <description>
+    Gitiles - Simple Git Repository Browser
+  </description>
+
+  <properties>
+    <!-- Should track Gerrit's jgitVersion fairly closely. -->
+    <jgitVersion>2.0.0.201206130900-r.129-ge63f1c9</jgitVersion>
+    <jettyVersion>7.5.2.v20111006</jettyVersion>
+  </properties>
+
+  <modules>
+    <module>gitiles-servlet</module>
+    <module>gitiles-war</module>
+  </modules>
+
+  <dependencyManagement>
+    <dependencies>
+      <dependency>
+        <groupId>com.google.guava</groupId>
+        <artifactId>guava</artifactId>
+        <version>13.0</version>
+      </dependency>
+
+      <dependency>
+        <groupId>com.google.template</groupId>
+        <artifactId>soy</artifactId>
+        <version>2011-22-12</version>
+      </dependency>
+
+      <dependency>
+        <groupId>org.eclipse.jgit</groupId>
+        <artifactId>org.eclipse.jgit</artifactId>
+        <version>${jgitVersion}</version>
+      </dependency>
+
+      <dependency>
+        <groupId>org.eclipse.jgit</groupId>
+        <artifactId>org.eclipse.jgit.http.server</artifactId>
+        <version>${jgitVersion}</version>
+      </dependency>
+
+      <dependency>
+        <groupId>org.eclipse.jgit</groupId>
+        <artifactId>org.eclipse.jgit.junit</artifactId>
+        <version>${jgitVersion}</version>
+        <exclusions>
+          <exclusion>
+            <groupId>org.eclipse.jgit</groupId>
+            <artifactId>org.eclipse.jgit</artifactId>
+          </exclusion>
+        </exclusions>
+      </dependency>
+
+      <dependency>
+        <groupId>org.apache.tomcat</groupId>
+        <artifactId>servlet-api</artifactId>
+        <version>6.0.29</version>
+      </dependency>
+
+      <dependency>
+        <groupId>junit</groupId>
+        <artifactId>junit</artifactId>
+        <version>3.8.1</version>
+      </dependency>
+
+      <dependency>
+        <groupId>org.slf4j</groupId>
+        <artifactId>slf4j-api</artifactId>
+        <version>1.6.1</version>
+      </dependency>
+
+      <dependency>
+        <groupId>joda-time</groupId>
+        <artifactId>joda-time</artifactId>
+        <version>2.1</version>
+      </dependency>
+
+      <dependency>
+        <groupId>com.google.code.gson</groupId>
+        <artifactId>gson</artifactId>
+        <version>2.1</version>
+      </dependency>
+
+      <dependency>
+        <groupId>org.apache.commons</groupId>
+        <artifactId>commons-lang3</artifactId>
+        <version>3.1</version>
+      </dependency>
+
+      <dependency>
+        <groupId>org.eclipse.jetty</groupId>
+        <artifactId>jetty-server</artifactId>
+        <version>${jettyVersion}</version>
+      </dependency>
+    </dependencies>
+  </dependencyManagement>
+
+  <build>
+    <pluginManagement>
+      <plugins>
+        <plugin>
+          <groupId>org.apache.maven.plugins</groupId>
+          <artifactId>maven-compiler-plugin</artifactId>
+          <version>2.3.2</version>
+          <configuration>
+            <source>1.6</source>
+            <target>1.6</target>
+            <encoding>UTF-8</encoding>
+          </configuration>
+        </plugin>
+
+        <plugin>
+          <groupId>org.apache.maven.plugins</groupId>
+          <artifactId>maven-jar-plugin</artifactId>
+          <configuration>
+            <archive>
+              <manifestEntries>
+                <Implementation-Title>Gitiles - ${project.artifactId}</Implementation-Title>
+                <Implementation-Version>${project.version}</Implementation-Version>
+                <Implementation-Vendor>Gitiles</Implementation-Vendor>
+                <Implementation-Vendor-Id>com.google.gitiles</Implementation-Vendor-Id>
+                <Implementation-Vendor-URL>https://gerrit.googlesource.com/gitiles/</Implementation-Vendor-URL>
+              </manifestEntries>
+            </archive>
+          </configuration>
+        </plugin>
+
+        <plugin>
+          <groupId>org.apache.maven.plugins</groupId>
+          <artifactId>maven-war-plugin</artifactId>
+          <version>2.1.1</version>
+        </plugin>
+
+        <plugin>
+          <groupId>org.apache.maven.plugins</groupId>
+          <artifactId>maven-source-plugin</artifactId>
+          <version>2.1.2</version>
+        </plugin>
+      </plugins>
+    </pluginManagement>
+
+    <plugins>
+      <plugin>
+        <groupId>org.mortbay.jetty</groupId>
+        <artifactId>jetty-maven-plugin</artifactId>
+        <version>${jettyVersion}</version>
+        <configuration>
+          <webApp>
+            <defaultsDescriptor>gitiles-war/webdefault.xml</defaultsDescriptor>
+          </webApp>
+          <war>gitiles-war/target/gitiles-war-${project.version}.war</war>
+        </configuration>
+        <!-- TODO(dborowitz): Separate execution with reloadable static
+          resources and templates. -->
+      </plugin>
+    </plugins>
+  </build>
+
+  <repositories>
+    <!-- For JGit and Soy snapshots. -->
+    <repository>
+      <id>gerrit-maven</id>
+      <url>https://gerrit-maven.commondatastorage.googleapis.com</url>
+    </repository>
+
+    <repository>
+      <id>jgit-repository</id>
+      <url>http://download.eclipse.org/jgit/maven</url>
+    </repository>
+
+    <repository>
+      <id>java.net-repository</id>
+      <url>http://download.java.net/maven/2/</url>
+    </repository>
+  </repositories>
+</project>