[tor-commits] [collector/release] Extend index.json by additional file meta data.

karsten at torproject.org karsten at torproject.org
Sat Nov 9 10:34:15 UTC 2019


commit 500b7c5ad3d94a0f0f8b8c7fdb110813895c25f0
Author: Karsten Loesing <karsten.loesing at gmx.net>
Date:   Wed Oct 30 21:56:41 2019 +0100

    Extend index.json by additional file meta data.
    
    Implements #31204.
---
 CHANGELOG.md                                       |  13 +
 build.xml                                          |   2 +-
 src/build                                          |   2 +-
 .../org/torproject/metrics/collector/conf/Key.java |   5 +-
 .../metrics/collector/indexer/CreateIndexJson.java | 678 +++++++++++++++++----
 .../metrics/collector/indexer/DirectoryNode.java   |  36 ++
 .../metrics/collector/indexer/FileNode.java        | 125 ++++
 .../metrics/collector/indexer/IndexNode.java       |  30 +
 .../metrics/collector/indexer/IndexerTask.java     | 225 +++++++
 src/main/resources/collector.properties            |  14 +-
 src/main/resources/create-tarballs.sh              |   2 +-
 .../metrics/collector/conf/ConfigurationTest.java  |   2 +-
 .../collector/indexer/CreateIndexJsonTest.java     | 522 ++++++++++++++++
 .../metrics/collector/indexer/IndexerTaskTest.java | 226 +++++++
 14 files changed, 1756 insertions(+), 126 deletions(-)

diff --git a/CHANGELOG.md b/CHANGELOG.md
index bb328ad..38754c4 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,5 +1,18 @@
 # Changes in version 1.??.? - 2019-??-??
 
+ * Medium changes
+   - Extend index.json by including descriptor types, first and last
+     publication timestamp, and SHA-256 file digest. Requires making
+     configuration changes in collector.properties:
+      1) IndexedPath is a new directory with subdirectories for
+         archived and recent descriptors,
+      2) ArchivePath and IndexPath are hard-wired to be subdirectories
+         of IndexedPath,
+      3) RecentPath must be set to be a subdirectory of IndexedPath,
+      4) ContribPath has disappeared, and
+      5) HtdocsPath is a new directory with files served by the web
+         server.
+
 
 # Changes in version 1.12.0 - 2019-10-18
 
diff --git a/build.xml b/build.xml
index 8276e63..019461d 100644
--- a/build.xml
+++ b/build.xml
@@ -12,7 +12,7 @@
   <property name="release.version" value="1.12.0-dev" />
   <property name="project-main-class" value="org.torproject.metrics.collector.Main" />
   <property name="name" value="collector"/>
-  <property name="metricslibversion" value="2.8.0" />
+  <property name="metricslibversion" value="2.9.0" />
   <property name="jarincludes" value="collector.properties logback.xml" />
 
   <patternset id="runtime" >
diff --git a/src/build b/src/build
index d82fff9..eb16cb3 160000
--- a/src/build
+++ b/src/build
@@ -1 +1 @@
-Subproject commit d82fff984634fe006ac7b0b102e7f48a52ca20d9
+Subproject commit eb16cb359db41722e6089bafb1e26808df4338df
diff --git a/src/main/java/org/torproject/metrics/collector/conf/Key.java b/src/main/java/org/torproject/metrics/collector/conf/Key.java
index 390feed..d59438b 100644
--- a/src/main/java/org/torproject/metrics/collector/conf/Key.java
+++ b/src/main/java/org/torproject/metrics/collector/conf/Key.java
@@ -18,13 +18,12 @@ public enum Key {
   RunOnce(Boolean.class),
   ExitlistUrl(URL.class),
   InstanceBaseUrl(String.class),
-  ArchivePath(Path.class),
-  ContribPath(Path.class),
+  IndexedPath(Path.class),
   RecentPath(Path.class),
   OutputPath(Path.class),
-  IndexPath(Path.class),
   StatsPath(Path.class),
   SyncPath(Path.class),
+  HtdocsPath(Path.class),
   RelaySources(SourceType[].class),
   BridgeSources(SourceType[].class),
   BridgePoolAssignmentsSources(SourceType[].class),
diff --git a/src/main/java/org/torproject/metrics/collector/indexer/CreateIndexJson.java b/src/main/java/org/torproject/metrics/collector/indexer/CreateIndexJson.java
index a40798e..15aa31d 100644
--- a/src/main/java/org/torproject/metrics/collector/indexer/CreateIndexJson.java
+++ b/src/main/java/org/torproject/metrics/collector/indexer/CreateIndexJson.java
@@ -1,73 +1,190 @@
-/* Copyright 2015--2018 The Tor Project
+/* Copyright 2015--2019 The Tor Project
  * See LICENSE for licensing information */
 
 package org.torproject.metrics.collector.indexer;
 
-import org.torproject.descriptor.index.DirectoryNode;
-import org.torproject.descriptor.index.FileNode;
-import org.torproject.descriptor.index.IndexNode;
-import org.torproject.descriptor.internal.FileType;
 import org.torproject.metrics.collector.conf.Configuration;
+import org.torproject.metrics.collector.conf.ConfigurationException;
 import org.torproject.metrics.collector.conf.Key;
 import org.torproject.metrics.collector.cron.CollecTorMain;
 
+import com.fasterxml.jackson.annotation.JsonAutoDetect;
+import com.fasterxml.jackson.annotation.JsonInclude;
+import com.fasterxml.jackson.annotation.PropertyAccessor;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.fasterxml.jackson.databind.PropertyNamingStrategy;
+import org.apache.commons.compress.compressors.bzip2.BZip2CompressorOutputStream;
+import org.apache.commons.compress.compressors.gzip.GzipCompressorOutputStream;
+import org.apache.commons.compress.compressors.xz.XZCompressorOutputStream;
 import org.slf4j.Logger;
 import org.slf4j.LoggerFactory;
 
-import java.io.BufferedWriter;
-import java.io.File;
-import java.io.FileOutputStream;
+import java.io.IOException;
 import java.io.InputStream;
-import java.io.OutputStreamWriter;
-import java.text.DateFormat;
-import java.text.SimpleDateFormat;
-import java.util.Locale;
+import java.io.OutputStream;
+import java.nio.file.FileVisitResult;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.nio.file.Paths;
+import java.nio.file.SimpleFileVisitor;
+import java.nio.file.StandardCopyOption;
+import java.nio.file.attribute.BasicFileAttributes;
+import java.time.Duration;
+import java.time.Instant;
+import java.time.ZoneOffset;
+import java.time.format.DateTimeFormatter;
+import java.time.temporal.TemporalAmount;
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.HashSet;
+import java.util.Map;
 import java.util.Properties;
+import java.util.Set;
+import java.util.SortedMap;
 import java.util.SortedSet;
-import java.util.TimeZone;
+import java.util.TreeMap;
 import java.util.TreeSet;
+import java.util.concurrent.ExecutionException;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
 
-/* Create a fresh index.json containing all directories and files in the
- * archive/ and recent/ directories.
+/**
+ * Create an index file called {@code index.json} containing metadata of all
+ * files in the {@code indexed/} directory and update the {@code htdocs/}
+ * directory to contain all files to be served via the web server.
  *
- * Note that if this ever takes longer than a few seconds, we'll have to
- * cache index parts of directories or files that haven't changed.
- * Example: if we parse include cryptographic hashes or @type information,
- * we'll likely have to do that. */
+ * <p>File metadata includes:</p>
+ * <ul>
+ *   <li>Path for downloading this file from the web server.</li>
+ *   <li>Size of the file in bytes.</li>
+ *   <li>Timestamp when the file was last modified.</li>
+ *   <li>Descriptor types as found in {@code @type} annotations of contained
+ *   descriptors.</li>
+ *   <li>Earliest and latest publication timestamp of contained
+ *   descriptors.</li>
+ *   <li>SHA-256 digest of the file.</li>
+ * </ul>
+ *
+ * <p>This class maintains its own working directory {@code htdocs/} with
+ * subdirectories like {@code htdocs/archive/} or {@code htdocs/recent/} and
+ * another subdirectory {@code htdocs/index/}. The first two subdirectories
+ * contain (hard) links created and deleted by this class, the third
+ * subdirectory contains the {@code index.json} file in uncompressed and
+ * compressed forms.</p>
+ *
+ * <p>The main reason for having the {@code htdocs/} directory is that indexing
+ * a large descriptor file can be time consuming. New or updated files in
+ * {@code indexed/} first need to be indexed before their metadata can be
+ * included in {@code index.json}. Another reason is that files removed from
+ * {@code indexed/} shall still be available for download for a limited period
+ * of time after disappearing from {@code index.json}.</p>
+ *
+ * <p>The reason for creating (hard) links in {@code htdocs/}, rather than
+ * copies, is that links do not consume additional disk space. All directories
+ * must be located on the same file system. Storing symbolic links in
+ * {@code htdocs/} would not have worked with replaced or deleted files in the
+ * original directories. Symbolic links in original directories are allowed as
+ * long as they target to the same file system.</p>
+ *
+ * <p>This class does not write, modify, or delete any files in the
+ * {@code indexed/} directory. At the same time it does not expect any other
+ * classes to write, modify, or delete contents in the {@code htdocs/}
+ * directory.</p>
+ */
 public class CreateIndexJson extends CollecTorMain {
 
+  /**
+   * Class logger.
+   */
   private static final Logger logger =
       LoggerFactory.getLogger(CreateIndexJson.class);
 
-  private static File indexJsonFile;
+  /**
+   * Delay between finding out that a file has been deleted and deleting its
+   * link.
+   */
+  private static final TemporalAmount deletionDelay = Duration.ofHours(2L);
 
-  private static String basePath;
+  /**
+   * Index tarballs with no more than this many threads at a time.
+   */
+  private static final int tarballIndexerThreads = 3;
 
-  private static File[] indexedDirectories;
+  /**
+   * Index flat files with no more than this many threads at a time.
+   */
+  private static final int flatFileIndexerThreads = 3;
 
-  private static final String dateTimePattern = "yyyy-MM-dd HH:mm";
+  /**
+   * Parser and formatter for all timestamps found in {@code index.json}.
+   */
+  private static DateTimeFormatter dateTimeFormatter = DateTimeFormatter
+      .ofPattern("uuuu-MM-dd HH:mm").withZone(ZoneOffset.UTC);
 
-  private static final Locale dateTimeLocale = Locale.US;
+  /**
+   * Object mapper for parsing and formatting {@code index.json} files.
+   */
+  private static ObjectMapper objectMapper = new ObjectMapper()
+      .setPropertyNamingStrategy(PropertyNamingStrategy.SNAKE_CASE)
+      .setSerializationInclusion(JsonInclude.Include.NON_EMPTY)
+      .setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.NONE)
+      .setVisibility(PropertyAccessor.FIELD, JsonAutoDetect.Visibility.ANY);
 
-  private static final TimeZone dateTimezone = TimeZone.getTimeZone("UTC");
+  /**
+   * Path to the {@code indexed/} directory.
+   */
+  private Path indexedPath;
 
-  private static String buildRevision = null;
+  /**
+   * Path to the {@code htdocs/} directory.
+   */
+  private Path htdocsPath;
 
-  /** Creates indexes of directories containing archived and recent
-   * descriptors and write index files to disk. */
-  public CreateIndexJson(Configuration conf) {
-    super(conf);
-    Properties buildProperties = new Properties();
-    try (InputStream is = getClass().getClassLoader()
-        .getResourceAsStream("collector.buildrevision.properties")) {
-      buildProperties.load(is);
-      buildRevision = buildProperties.getProperty("collector.build.revision",
-          null);
-    } catch (Exception ex) {
-      // This doesn't hamper the index creation: only log a warning.
-      logger.warn("No build revision available.", ex);
-      buildRevision = null;
-    }
+  /**
+   * Path to the uncompressed {@code index.json} file.
+   */
+  private Path indexJsonPath;
+
+  /**
+   * Base URL of all resources included in {@code index.json}.
+   */
+  private String basePathString;
+
+  /**
+   * Git revision of this software to be included in {@code index.json} or
+   * omitted if unknown.
+   */
+  private String buildRevisionString;
+
+  /**
+   * Index containing metadata of files in {@code indexed/}, including new or
+   * updated files that still need to be indexed and deleted files that are
+   * still linked in {@code htdocs/}.
+   *
+   * <p>This map is initialized by reading the last known {@code index.json}
+   * file and remains available in memory between executions until shutdown.</p>
+   */
+  private SortedMap<Path, FileNode> index;
+
+  /**
+   * Executor for indexing tarballs.
+   */
+  private ExecutorService tarballsExecutor
+      = Executors.newFixedThreadPool(tarballIndexerThreads);
+
+  /**
+   * Executor for indexing flat files (non-tarballs).
+   */
+  private ExecutorService flatFilesExecutor
+      = Executors.newFixedThreadPool(flatFileIndexerThreads);
+
+  /**
+   * Initialize this class with the given {@code configuration}.
+   *
+   * @param configuration Configuration values.
+   */
+  public CreateIndexJson(Configuration configuration) {
+    super(configuration);
   }
 
   @Override
@@ -80,96 +197,433 @@ public class CreateIndexJson extends CollecTorMain {
     return "IndexJson";
   }
 
+
+  /**
+   * Run the indexer by (1) adding new files from {@code indexed/} to the index,
+   * (2) adding old files from {@code htdocs/} for which only links exist to the
+   * index, (3) scheduling new tasks and updating links in {@code htdocs/} to
+   * reflect what's contained in the in-memory index, and (4) writing new
+   * uncompressed and compressed {@code index.json} files to disk.
+   */
   @Override
-  protected void startProcessing() {
+  public void startProcessing() {
+    this.startProcessing(Instant.now());
+  }
+
+  /**
+   * Helper method to {@link #startProcessing()} that accepts the current
+   * execution time and which is used by tests.
+   *
+   * @param now Current execution time.
+   */
+  protected void startProcessing(Instant now) {
+    try {
+      this.basePathString = this.config.getProperty(Key.InstanceBaseUrl.name());
+      this.indexedPath = config.getPath(Key.IndexedPath);
+      this.htdocsPath = config.getPath(Key.HtdocsPath);
+    } catch (ConfigurationException e) {
+      logger.error("Unable to read one or more configuration values. Not "
+          + "indexing in this execution.", e);
+    }
+    this.buildRevisionString = this.obtainBuildRevision();
+    this.indexJsonPath = this.htdocsPath
+        .resolve(Paths.get("index", "index.json"));
     try {
-      indexJsonFile = new File(config.getPath(Key.IndexPath).toFile(),
-          "index.json");
-      basePath = config.getProperty(Key.InstanceBaseUrl.name());
-      indexedDirectories = new File[] {
-          config.getPath(Key.ArchivePath).toFile(),
-          config.getPath(Key.ContribPath).toFile(),
-          config.getPath(Key.RecentPath).toFile() };
-      writeIndex(indexDirectories());
-    } catch (Exception e) {
-      logger.error("Cannot run index creation: {}", e.getMessage(), e);
-      throw new RuntimeException(e);
+      this.prepareHtdocsDirectory();
+      if (null == this.index) {
+        logger.info("Reading index.json file from last execution.");
+        this.index = this.readIndex();
+      }
+      logger.info("Going through indexed/ and adding new files to the index.");
+      this.addNewFilesToIndex(this.indexedPath);
+      logger.info("Going through htdocs/ and adding links to deleted files to "
+          + "the index.");
+      this.addOldLinksToIndex();
+      logger.info("Going through the index, scheduling tasks, and updating "
+          + "links.");
+      this.scheduleTasksAndUpdateLinks(now);
+      logger.info("Writing uncompressed and compressed index.json files to "
+          + "disk.");
+      this.writeIndex(this.index, now);
+      logger.info("Pausing until next index update run.");
+    } catch (IOException e) {
+      logger.error("I/O error while updating index.json files. Trying again in "
+          + "the next execution.", e);
     }
   }
 
-  private static DateFormat dateTimeFormat;
-
-  static {
-    dateTimeFormat = new SimpleDateFormat(dateTimePattern,
-        dateTimeLocale);
-    dateTimeFormat.setLenient(false);
-    dateTimeFormat.setTimeZone(dateTimezone);
+  /**
+   * Prepare the {@code htdocs/} directory by checking whether all required
+   * subdirectories exist and by creating them if not.
+   *
+   * @throws IOException Thrown if one or more directories could not be created.
+   */
+  private void prepareHtdocsDirectory() throws IOException {
+    for (Path requiredPath : new Path[] {
+        this.htdocsPath,
+        this.indexJsonPath.getParent() }) {
+      if (!Files.exists(requiredPath)) {
+        Files.createDirectories(requiredPath);
+      }
+    }
   }
 
-  private IndexNode indexDirectories() {
-    SortedSet<DirectoryNode> directoryNodes = new TreeSet<>();
-    logger.trace("indexing: {} {}", indexedDirectories[0],
-        indexedDirectories[1]);
-    for (File directory : indexedDirectories) {
-      if (directory.exists() && directory.isDirectory()) {
-        DirectoryNode dn = indexDirectory(directory);
-        if (null != dn) {
-          directoryNodes.add(dn);
+  /**
+   * Read the {@code index.json} file written by the previous execution and
+   * populate our index with its contents, or leave the index empty if this is
+   * the first execution and that file does not yet exist.
+   *
+   * @return Index read from disk, or empty map if {@code index.json} does not
+   *     exist.
+   */
+  private SortedMap<Path, FileNode> readIndex() throws IOException {
+    SortedMap<Path, FileNode> index = new TreeMap<>();
+    if (Files.exists(this.indexJsonPath)) {
+      IndexNode indexNode = objectMapper.readValue(
+          Files.newInputStream(this.indexJsonPath), IndexNode.class);
+      SortedMap<Path, DirectoryNode> directoryNodes = new TreeMap<>();
+      directoryNodes.put(Paths.get(""), indexNode);
+      while (!directoryNodes.isEmpty()) {
+        Path directoryPath = directoryNodes.firstKey();
+        DirectoryNode directoryNode = directoryNodes.remove(directoryPath);
+        if (null != directoryNode.files) {
+          for (FileNode fileNode : directoryNode.files) {
+            Path filePath = this.indexedPath.resolve(directoryPath)
+                .resolve(Paths.get(fileNode.path));
+            index.put(filePath, fileNode);
+          }
         }
+        if (null != directoryNode.directories) {
+          boolean isRootDirectory = directoryNode == indexNode;
+          for (DirectoryNode subdirectoryNode : directoryNode.directories) {
+            Path subdirectoryPath = isRootDirectory
+                ? Paths.get(subdirectoryNode.path)
+                : directoryPath.resolve(Paths.get(subdirectoryNode.path));
+            directoryNodes.put(subdirectoryPath, subdirectoryNode);
+          }
+        }
+      }
+    }
+    return index;
+  }
+
+  /**
+   * Obtain and return the build revision string that was generated during the
+   * build process with {@code git rev-parse --short HEAD} and written to
+   * {@code collector.buildrevision.properties}, or return {@code null} if the
+   * build revision string cannot be obtained.
+   *
+   * @return Build revision string.
+   */
+  protected String obtainBuildRevision() {
+    String buildRevision = null;
+    Properties buildProperties = new Properties();
+    String propertiesFile = "collector.buildrevision.properties";
+    try (InputStream is = getClass().getClassLoader()
+        .getResourceAsStream(propertiesFile)) {
+      if (null == is) {
+        logger.warn("File {}, which is supposed to contain the build revision "
+            + "string, does not exist in our class path. Writing index.json "
+            + "without the \"build_revision\" field.", propertiesFile);
+        return null;
       }
+      buildProperties.load(is);
+      buildRevision = buildProperties.getProperty(
+          "collector.build.revision", null);
+    } catch (IOException e) {
+      logger.warn("I/O error while trying to obtain build revision string. "
+          + "Writing index.json without the \"build_revision\" field.");
     }
-    return new IndexNode(dateTimeFormat.format(
-        System.currentTimeMillis()), buildRevision, basePath, null,
-        directoryNodes);
-  }
-
-  private DirectoryNode indexDirectory(File directory) {
-    SortedSet<FileNode> fileNodes = new TreeSet<>();
-    SortedSet<DirectoryNode> directoryNodes = new TreeSet<>();
-    logger.trace("indexing: {}", directory);
-    File[] fileList = directory.listFiles();
-    if (null == fileList) {
-      logger.warn("Indexing dubious directory: {}", directory);
-      return null;
+    return buildRevision;
+  }
+
+  /**
+   * Walk the given file tree and add all previously unknown files to the
+   * in-memory index (except for files starting with "." or ending with ".tmp").
+   *
+   * @param path File tree to walk.
+   */
+  private void addNewFilesToIndex(Path path) throws IOException {
+    if (!Files.exists(path)) {
+      return;
     }
-    for (File fileOrDirectory : fileList) {
-      if (fileOrDirectory.getName().startsWith(".")
-          || fileOrDirectory.getName().endsWith(".tmp")) {
-        continue;
+    Files.walkFileTree(path, new SimpleFileVisitor<Path>() {
+      @Override
+      public FileVisitResult visitFile(Path filePath,
+          BasicFileAttributes basicFileAttributes) {
+        if (!filePath.toString().startsWith(".")
+            && !filePath.toString().endsWith(".tmp")) {
+          index.putIfAbsent(filePath, new FileNode());
+        }
+        return FileVisitResult.CONTINUE;
+      }
+    });
+  }
+
+  /**
+   * Walk the file tree of the {@code htdocs/} directory and add all previously
+   * unknown links to the in-memory index to ensure their deletion when they're
+   * known to be deleted from their original directories.
+   */
+  private void addOldLinksToIndex() throws IOException {
+    Path htdocsIndexPath = this.indexJsonPath.getParent();
+    Files.walkFileTree(this.htdocsPath, new SimpleFileVisitor<Path>() {
+      @Override
+      public FileVisitResult visitFile(Path linkPath,
+          BasicFileAttributes basicFileAttributes) {
+        if (!linkPath.startsWith(htdocsIndexPath)) {
+          Path filePath = indexedPath.resolve(htdocsPath.relativize(linkPath));
+          index.putIfAbsent(filePath, new FileNode());
+        }
+        return FileVisitResult.CONTINUE;
       }
-      if (fileOrDirectory.isFile()) {
-        fileNodes.add(indexFile(fileOrDirectory));
+    });
+  }
+
+  /**
+   * Go through the index, schedule tasks to index files, and update links.
+   *
+   * @throws IOException Thrown if an I/O exception occurs while creating or
+   *     deleting links.
+   */
+  private void scheduleTasksAndUpdateLinks(Instant now) throws IOException {
+    int queuedIndexerTasks = 0;
+    Map<Path, FileNode> indexingResults = new HashMap<>();
+    SortedSet<Path> filesToIndex = new TreeSet<>();
+    Map<Path, Path> linksToCreate = new HashMap<>();
+    Set<FileNode> linksToMarkForDeletion = new HashSet<>();
+    Map<Path, Path> linksToDelete = new HashMap<>();
+    for (Map.Entry<Path, FileNode> e : this.index.entrySet()) {
+      Path filePath = e.getKey();
+      Path linkPath = this.htdocsPath
+          .resolve(this.indexedPath.relativize(filePath));
+      FileNode fileNode = e.getValue();
+      if (Files.exists(filePath)) {
+        if (null != fileNode.indexerResult) {
+          if (!fileNode.indexerResult.isDone()) {
+            /* This file is currently being indexed, so we should just skip it
+             * and wait until the indexer is done. */
+            queuedIndexerTasks++;
+            continue;
+          }
+          try {
+            /* Indexing is done, obtain index results. */
+            fileNode = fileNode.indexerResult.get();
+            indexingResults.put(filePath, fileNode);
+          } catch (InterruptedException | ExecutionException ex) {
+            /* Clear index result, so that we can give this file another try
+             * next time. */
+            fileNode.indexerResult = null;
+          }
+        }
+        String originalLastModified = dateTimeFormatter
+            .format(Files.getLastModifiedTime(filePath).toInstant());
+        if (!originalLastModified.equals(fileNode.lastModified)) {
+          /* We either don't have any index results for this file, or we only
+           * have index results for an older version of this file. */
+          filesToIndex.add(filePath);
+        } else if (!Files.exists(linkPath)) {
+          /* We do have index results, but we don't have a link yet, so we're
+           * going to create a link. */
+          linksToCreate.put(linkPath, filePath);
+        } else {
+          String linkLastModified = dateTimeFormatter
+              .format(Files.getLastModifiedTime(linkPath).toInstant());
+          if (!linkLastModified.equals(fileNode.lastModified)) {
+            /* We do have index results plus a link to an older version of this
+             * file, so we'll have to update the link. */
+            linksToCreate.put(linkPath, filePath);
+          }
+        }
       } else {
-        DirectoryNode dn = indexDirectory(fileOrDirectory);
-        if (null != dn) {
-          directoryNodes.add(dn);
+        if (null == fileNode.markedForDeletion) {
+          /* We're noticing just now that the file doesn't exist anymore, so
+           * we're going to mark it for deletion but not deleting the link right
+           * away. */
+          linksToMarkForDeletion.add(fileNode);
+        } else if (fileNode.markedForDeletion
+            .isBefore(now.minus(deletionDelay))) {
+          /* The file doesn't exist anymore, and we found out long enough ago,
+           * so we can now go ahead and delete the link. */
+          linksToDelete.put(linkPath, filePath);
+        }
+      }
+    }
+    if (queuedIndexerTasks > 0) {
+      logger.info("Counting {} file(s) being currently indexed or in the "
+          + "queue.", queuedIndexerTasks);
+    }
+    this.updateIndex(indexingResults);
+    this.scheduleTasks(filesToIndex);
+    this.createLinks(linksToCreate);
+    this.markForDeletion(linksToMarkForDeletion, now);
+    this.deleteLinks(linksToDelete);
+  }
+
+  /**
+   * Update index with index results.
+   */
+  private void updateIndex(Map<Path, FileNode> indexResults) {
+    if (!indexResults.isEmpty()) {
+      logger.info("Updating {} index entries with index results.",
+          indexResults.size());
+      this.index.putAll(indexResults);
+    }
+  }
+
+  /**
+   * Schedule indexing the given set of descriptor files, using different queues
+   * for tarballs and flat files.
+   *
+   * @param filesToIndex Paths to descriptor files to index.
+   */
+  private void scheduleTasks(SortedSet<Path> filesToIndex) {
+    if (!filesToIndex.isEmpty()) {
+      logger.info("Scheduling {} indexer task(s).", filesToIndex.size());
+      for (Path fileToIndex : filesToIndex) {
+        IndexerTask indexerTask = this.createIndexerTask(fileToIndex);
+        if (fileToIndex.getFileName().toString().endsWith(".tar.xz")) {
+          this.index.get(fileToIndex).indexerResult
+              = this.tarballsExecutor.submit(indexerTask);
+        } else {
+          this.index.get(fileToIndex).indexerResult
+              = this.flatFilesExecutor.submit(indexerTask);
         }
       }
     }
-    return new DirectoryNode(
-        directory.getName(), fileNodes.isEmpty() ? null : fileNodes,
-        directoryNodes.isEmpty() ? null : directoryNodes);
-  }
-
-  private FileNode indexFile(File file) {
-    return new FileNode(file.getName(), file.length(),
-        dateTimeFormat.format(file.lastModified()));
-  }
-
-  private void writeIndex(IndexNode indexNode) throws Exception {
-    indexJsonFile.getParentFile().mkdirs();
-    String indexNodeString = IndexNode.makeJsonString(indexNode);
-    for (String filename : new String[] {indexJsonFile.toString(),
-        indexJsonFile + ".gz", indexJsonFile + ".xz", indexJsonFile + ".bz2"}) {
-      FileType type = FileType.valueOf(
-          filename.substring(filename.lastIndexOf(".") + 1).toUpperCase());
-      try (BufferedWriter bufferedWriter
-          = new BufferedWriter(new OutputStreamWriter(type.outputStream(
-          new FileOutputStream(filename))))) {
-        bufferedWriter.write(indexNodeString);
+  }
+
+  /**
+   * Create an indexer task for indexing the given file.
+   *
+   * <p>The reason why this is a separate method is that it can be overriden by
+   * tests that don't actually want to index files but instead provide their own
+   * index results.</p>
+   *
+   * @param fileToIndex File to index.
+   * @return Indexer task.
+   */
+  protected IndexerTask createIndexerTask(Path fileToIndex) {
+    return new IndexerTask(fileToIndex);
+  }
+
+  /**
+   * Create links in {@code htdocs/}, including all necessary parent
+   * directories.
+   *
+   * @param linksToCreate Map of links to be created with keys being link paths
+   *     and values being original file paths.
+   * @throws IOException Thrown if an I/O error occurs.
+   */
+  private void createLinks(Map<Path, Path> linksToCreate) throws IOException {
+    if (!linksToCreate.isEmpty()) {
+      logger.info("Creating {} new link(s).", linksToCreate.size());
+      for (Map.Entry<Path, Path> e : linksToCreate.entrySet()) {
+        Path linkPath = e.getKey();
+        Path originalPath = e.getValue();
+        Files.createDirectories(linkPath.getParent());
+        Files.deleteIfExists(linkPath);
+        Files.createLink(linkPath, originalPath);
+      }
+    }
+  }
+
+  /**
+   * Mark the given links for deletion in the in-memory index.
+   *
+   * @param linksToMarkForDeletion Files to be marked for deletion.
+   */
+  private void markForDeletion(Set<FileNode> linksToMarkForDeletion,
+      Instant now) {
+    if (!linksToMarkForDeletion.isEmpty()) {
+      logger.info("Marking {} old link(s) for deletion.",
+          linksToMarkForDeletion.size());
+      for (FileNode fileNode : linksToMarkForDeletion) {
+        fileNode.markedForDeletion = now;
+      }
+    }
+  }
+
+  /**
+   * Delete the given links from {@code htdocs/}.
+   *
+   * @param linksToDelete Map of links to be deleted with keys being link paths
+   *     and values being original file paths.
+   * @throws IOException Thrown if an I/O error occurs.
+   */
+  private void deleteLinks(Map<Path, Path> linksToDelete) throws IOException {
+    if (!linksToDelete.isEmpty()) {
+      logger.info("Deleting {} old link(s).", linksToDelete.size());
+      for (Map.Entry<Path, Path> e : linksToDelete.entrySet()) {
+        Path linkPath = e.getKey();
+        Path originalPath = e.getValue();
+        Files.deleteIfExists(linkPath);
+        index.remove(originalPath);
       }
     }
   }
 
+  /**
+   * Write the in-memory index to {@code index.json} and its compressed
+   * variants, but exclude files that have not yet been indexed or that are
+   * marked for deletion.
+   *
+   * @throws IOException Thrown if an I/O error occurs while writing files.
+   */
+  private void writeIndex(SortedMap<Path, FileNode> index,
+      Instant now) throws IOException {
+    IndexNode indexNode = new IndexNode();
+    indexNode.indexCreated = dateTimeFormatter.format(now);
+    indexNode.buildRevision = this.buildRevisionString;
+    indexNode.path = this.basePathString;
+    SortedMap<Path, DirectoryNode> directoryNodes = new TreeMap<>();
+    for (Map.Entry<Path, FileNode> indexEntry : index.entrySet()) {
+      Path filePath = this.indexedPath.relativize(indexEntry.getKey());
+      FileNode fileNode = indexEntry.getValue();
+      if (null == fileNode.lastModified || null != fileNode.markedForDeletion) {
+        /* Skip unindexed or deleted files. */
+        continue;
+      }
+      Path directoryPath = null;
+      DirectoryNode parentDirectoryNode = indexNode;
+      if (null != filePath.getParent()) {
+        for (Path pathPart : filePath.getParent()) {
+          directoryPath = null == directoryPath ? pathPart
+              : directoryPath.resolve(pathPart);
+          DirectoryNode directoryNode = directoryNodes.get(directoryPath);
+          if (null == directoryNode) {
+            directoryNode = new DirectoryNode();
+            directoryNode.path = pathPart.toString();
+            if (null == parentDirectoryNode.directories) {
+              parentDirectoryNode.directories = new ArrayList<>();
+            }
+            parentDirectoryNode.directories.add(directoryNode);
+            directoryNodes.put(directoryPath, directoryNode);
+          }
+          parentDirectoryNode = directoryNode;
+        }
+      }
+      if (null == parentDirectoryNode.files) {
+        parentDirectoryNode.files = new ArrayList<>();
+      }
+      parentDirectoryNode.files.add(fileNode);
+    }
+    Path htdocsIndexPath = this.indexJsonPath.getParent();
+    try (OutputStream uncompressed
+            = Files.newOutputStream(htdocsIndexPath.resolve(".index.json.tmp"));
+        OutputStream bz2Compressed = new BZip2CompressorOutputStream(
+            Files.newOutputStream(htdocsIndexPath.resolve("index.json.bz2")));
+        OutputStream gzCompressed = new GzipCompressorOutputStream(
+            Files.newOutputStream(htdocsIndexPath.resolve("index.json.gz")));
+        OutputStream xzCompressed = new XZCompressorOutputStream(
+            Files.newOutputStream(htdocsIndexPath.resolve("index.json.xz")))) {
+      objectMapper.writeValue(uncompressed, indexNode);
+      objectMapper.writeValue(bz2Compressed, indexNode);
+      objectMapper.writeValue(gzCompressed, indexNode);
+      objectMapper.writeValue(xzCompressed, indexNode);
+    }
+    Files.move(htdocsIndexPath.resolve(".index.json.tmp"), this.indexJsonPath,
+        StandardCopyOption.REPLACE_EXISTING);
+  }
 }
 
diff --git a/src/main/java/org/torproject/metrics/collector/indexer/DirectoryNode.java b/src/main/java/org/torproject/metrics/collector/indexer/DirectoryNode.java
new file mode 100644
index 0000000..a369d08
--- /dev/null
+++ b/src/main/java/org/torproject/metrics/collector/indexer/DirectoryNode.java
@@ -0,0 +1,36 @@
+/* Copyright 2019 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.metrics.collector.indexer;
+
+import com.fasterxml.jackson.annotation.JsonProperty;
+import com.fasterxml.jackson.annotation.JsonPropertyOrder;
+
+import java.util.List;
+
+/**
+ * Directory node in {@code index.json} which is discarded after reading and
+ * re-created before writing that file.
+ */
+ at JsonPropertyOrder({ "path", "files", "directories" })
+class DirectoryNode {
+
+  /**
+   * Relative path of the directory.
+   */
+  @JsonProperty("path")
+  String path;
+
+  /**
+   * List of file objects of files available from this directory.
+   */
+  @JsonProperty("files")
+  List<FileNode> files;
+
+  /**
+   * List of directory objects of directories available from this directory.
+   */
+  @JsonProperty("directories")
+  List<DirectoryNode> directories;
+}
+
diff --git a/src/main/java/org/torproject/metrics/collector/indexer/FileNode.java b/src/main/java/org/torproject/metrics/collector/indexer/FileNode.java
new file mode 100644
index 0000000..c007196
--- /dev/null
+++ b/src/main/java/org/torproject/metrics/collector/indexer/FileNode.java
@@ -0,0 +1,125 @@
+/* Copyright 2019 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.metrics.collector.indexer;
+
+import com.fasterxml.jackson.annotation.JsonIgnore;
+import com.fasterxml.jackson.annotation.JsonProperty;
+import com.fasterxml.jackson.annotation.JsonPropertyOrder;
+
+import java.time.Instant;
+import java.util.SortedSet;
+import java.util.TreeSet;
+import java.util.concurrent.Future;
+
+/**
+ * File node in {@code index.json}, also used for storing volatile metadata
+ * like whether a descriptor file is currently being indexed or whether its
+ * link in {@code htdocs/} is marked for deletion.
+ */
+ at JsonPropertyOrder({ "path", "size", "last_modified", "types",
+    "first_published", "last_published", "sha256" })
+class FileNode {
+
+  /**
+   * Relative path of the file.
+   */
+  @JsonProperty("path")
+  String path;
+
+  /**
+   * Size of the file in bytes.
+   */
+  @JsonProperty("size")
+  Long size;
+
+  /**
+   * Timestamp when the file was last modified using pattern
+   * {@code "YYYY-MM-DD HH:MM"} in the UTC timezone.
+   */
+  @JsonProperty("last_modified")
+  String lastModified;
+
+  /**
+   * Descriptor types as found in {@code @type} annotations of contained
+   * descriptors.
+   */
+  @JsonProperty("types")
+  SortedSet<String> types;
+
+  /**
+   * Earliest publication timestamp of contained descriptors using pattern
+   * {@code "YYYY-MM-DD HH:MM"} in the UTC timezone.
+   */
+  @JsonProperty("first_published")
+  String firstPublished;
+
+  /**
+   * Latest publication timestamp of contained descriptors using pattern
+   * {@code "YYYY-MM-DD HH:MM"} in the UTC timezone.
+   */
+  @JsonProperty("last_published")
+  String lastPublished;
+
+  /**
+   * SHA-256 digest of this file.
+   */
+  @JsonProperty("sha256")
+  String sha256;
+
+  /**
+   * Indexer result that will be available as soon as the indexer has completed
+   * its task.
+   */
+  @JsonIgnore
+  Future<FileNode> indexerResult;
+
+  /**
+   * Timestamp when this file was first not found anymore in {@code indexed/},
+   * used to keep the link in {@code htdocs/} around for another 2 hours before
+   * deleting it, too.
+   *
+   * <p>This field is ignored when writing {@code index.json}, because it's an
+   * internal detail that nobody else cares about. The effect is that links
+   * might be around for longer than 2 hours in case of a restart, which seems
+   * acceptable.</p>
+   */
+  @JsonIgnore
+  Instant markedForDeletion;
+
+  /**
+   * Create and return a {@link FileNode} instance with the given values.
+   *
+   * @param path Relative path of the file.
+   * @param size Size of the file in bytes.
+   * @param lastModified Timestamp when the file was last modified using pattern
+   *     {@code "YYYY-MM-DD HH:MM"} in the UTC timezone.
+   * @param types Descriptor types as found in {@code @type} annotations of
+   *     contained descriptors.
+   * @param firstPublished Earliest publication timestamp of contained
+   *     descriptors using pattern {@code "YYYY-MM-DD HH:MM"} in the UTC
+   *     timezone.
+   * @param lastPublished Latest publication timestamp of contained descriptors
+   *     using pattern {@code "YYYY-MM-DD HH:MM"} in the UTC timezone.
+   * @param sha256 SHA-256 digest of this file.
+   *
+   * @return {@link FileNode} instance with the given values.
+   */
+  static FileNode of(String path, Long size, String lastModified,
+      Iterable<String> types, String firstPublished, String lastPublished,
+      String sha256) {
+    FileNode fileNode = new FileNode();
+    fileNode.path = path;
+    fileNode.size = size;
+    fileNode.lastModified = lastModified;
+    fileNode.types = new TreeSet<>();
+    for (String type : types) {
+      fileNode.types.add(type);
+    }
+    fileNode.firstPublished = firstPublished;
+    fileNode.lastPublished = lastPublished;
+    fileNode.sha256 = sha256;
+    return fileNode;
+  }
+}
+
diff --git a/src/main/java/org/torproject/metrics/collector/indexer/IndexNode.java b/src/main/java/org/torproject/metrics/collector/indexer/IndexNode.java
new file mode 100644
index 0000000..8b7a46b
--- /dev/null
+++ b/src/main/java/org/torproject/metrics/collector/indexer/IndexNode.java
@@ -0,0 +1,30 @@
+/* Copyright 2019 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.metrics.collector.indexer;
+
+import com.fasterxml.jackson.annotation.JsonProperty;
+import com.fasterxml.jackson.annotation.JsonPropertyOrder;
+
+/**
+ * Root node in {@code index.json} containing additional information about index
+ * creation time or Git revision used for creating it.
+ */
+ at JsonPropertyOrder({ "index_created", "build_revision", "path", "files",
+    "directories" })
+class IndexNode extends DirectoryNode {
+
+  /**
+   * Timestamp when this index was created using pattern
+   * {@code "YYYY-MM-DD HH:MM"} in the UTC timezone.
+   */
+  @JsonProperty("index_created")
+  String indexCreated;
+
+  /**
+   * Git revision of this software.
+   */
+  @JsonProperty("build_revision")
+  String buildRevision;
+}
+
diff --git a/src/main/java/org/torproject/metrics/collector/indexer/IndexerTask.java b/src/main/java/org/torproject/metrics/collector/indexer/IndexerTask.java
new file mode 100644
index 0000000..03c750b
--- /dev/null
+++ b/src/main/java/org/torproject/metrics/collector/indexer/IndexerTask.java
@@ -0,0 +1,225 @@
+/* Copyright 2019 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.metrics.collector.indexer;
+
+import org.torproject.descriptor.BandwidthFile;
+import org.torproject.descriptor.BridgeNetworkStatus;
+import org.torproject.descriptor.BridgePoolAssignment;
+import org.torproject.descriptor.BridgedbMetrics;
+import org.torproject.descriptor.Descriptor;
+import org.torproject.descriptor.DescriptorSourceFactory;
+import org.torproject.descriptor.DirectoryKeyCertificate;
+import org.torproject.descriptor.ExitList;
+import org.torproject.descriptor.ExtraInfoDescriptor;
+import org.torproject.descriptor.Microdescriptor;
+import org.torproject.descriptor.RelayDirectory;
+import org.torproject.descriptor.RelayNetworkStatus;
+import org.torproject.descriptor.RelayNetworkStatusConsensus;
+import org.torproject.descriptor.RelayNetworkStatusVote;
+import org.torproject.descriptor.ServerDescriptor;
+import org.torproject.descriptor.SnowflakeStats;
+import org.torproject.descriptor.TorperfResult;
+import org.torproject.descriptor.UnparseableDescriptor;
+import org.torproject.descriptor.WebServerAccessLog;
+
+import org.apache.commons.codec.binary.Base64;
+import org.apache.commons.codec.digest.DigestUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+import java.io.IOException;
+import java.io.InputStream;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.time.Instant;
+import java.time.LocalDateTime;
+import java.time.ZoneOffset;
+import java.time.format.DateTimeFormatter;
+import java.util.SortedSet;
+import java.util.TreeSet;
+import java.util.concurrent.Callable;
+
+/**
+ * Callable task that indexes a given descriptor file.
+ */
+class IndexerTask implements Callable<FileNode> {
+
+  /**
+   * Class logger.
+   */
+  private static final Logger logger
+      = LoggerFactory.getLogger(IndexerTask.class);
+
+  /**
+   * Formatter for all timestamps found in {@code index.json}.
+   */
+  private static DateTimeFormatter dateTimeFormatter = DateTimeFormatter
+      .ofPattern("uuuu-MM-dd HH:mm").withZone(ZoneOffset.UTC);
+
+  /**
+   * Path to the descriptor file to index.
+   */
+  private Path path;
+
+  /**
+   * Index results object, which starts out empty and gets populated as indexing
+   * proceeds.
+   */
+  private FileNode indexResult;
+
+  /**
+   * Create a new instance to parse the given descriptor file, but don't start
+   * parsing just yet.
+   *
+   * @param path Descriptor file to index.
+   */
+  IndexerTask(Path path) {
+    this.path = path;
+  }
+
+  /**
+   * Index the given file and return index results when done.
+   *
+   * @return Index results.
+   * @throws IOException Thrown if an I/O error occurs.
+   */
+  @Override
+  public FileNode call() throws IOException {
+    this.indexResult = new FileNode();
+    this.requestBasicFileAttributes();
+    this.computeFileDigest();
+    this.parseDescriptorFile();
+    return this.indexResult;
+  }
+
+  /**
+   * Request and store basic file attributes like file name, last-modified time,
+   * and size.
+   *
+   * @throws IOException Thrown if an I/O error occurs.
+   */
+  private void requestBasicFileAttributes() throws IOException {
+    this.indexResult.path = this.path.getFileName().toString();
+    this.indexResult.lastModified = dateTimeFormatter
+        .format(Files.getLastModifiedTime(this.path).toInstant());
+    this.indexResult.size = Files.size(this.path);
+  }
+
+  /**
+   * Compute and store the file's SHA-256 digest.
+   *
+   * @throws IOException Thrown if an I/O error occurs.
+   */
+  private void computeFileDigest() throws IOException {
+    try (InputStream stream = Files.newInputStream(this.path)) {
+      this.indexResult.sha256
+          = Base64.encodeBase64String(DigestUtils.sha256(stream));
+    }
+  }
+
+  /**
+   * Parse the descriptor file to extract contained descriptor types and first
+   * and last published time.
+   */
+  private void parseDescriptorFile() {
+    Long firstPublishedMillis = null;
+    Long lastPublishedMillis = null;
+    this.indexResult.types = new TreeSet<>();
+    SortedSet<String> unknownDescriptorSubclasses = new TreeSet<>();
+    for (Descriptor descriptor : DescriptorSourceFactory
+        .createDescriptorReader().readDescriptors(this.path.toFile())) {
+      if (descriptor instanceof UnparseableDescriptor) {
+        /* Skip unparseable descriptor. */
+        continue;
+      }
+      for (String annotation : descriptor.getAnnotations()) {
+        if (annotation.startsWith("@type ")) {
+          this.indexResult.types.add(annotation.substring(6));
+        }
+      }
+      Long publishedMillis;
+      if (descriptor instanceof BandwidthFile) {
+        BandwidthFile bandwidthFile = (BandwidthFile) descriptor;
+        LocalDateTime fileCreatedOrTimestamp
+            = bandwidthFile.fileCreated().isPresent()
+            ? bandwidthFile.fileCreated().get()
+            : bandwidthFile.timestamp();
+        publishedMillis = fileCreatedOrTimestamp
+            .toInstant(ZoneOffset.UTC).toEpochMilli();
+      } else if (descriptor instanceof BridgeNetworkStatus) {
+        publishedMillis = ((BridgeNetworkStatus) descriptor)
+            .getPublishedMillis();
+      } else if (descriptor instanceof BridgePoolAssignment) {
+        publishedMillis = ((BridgePoolAssignment) descriptor)
+            .getPublishedMillis();
+      } else if (descriptor instanceof BridgedbMetrics) {
+        publishedMillis = ((BridgedbMetrics) descriptor)
+            .bridgedbMetricsEnd().toInstant(ZoneOffset.UTC).toEpochMilli();
+      } else if (descriptor instanceof DirectoryKeyCertificate) {
+        publishedMillis = ((DirectoryKeyCertificate) descriptor)
+            .getDirKeyPublishedMillis();
+      } else if (descriptor instanceof ExitList) {
+        publishedMillis = ((ExitList) descriptor)
+            .getDownloadedMillis();
+      } else if (descriptor instanceof ExtraInfoDescriptor) {
+        publishedMillis = ((ExtraInfoDescriptor) descriptor)
+            .getPublishedMillis();
+      } else if (descriptor instanceof Microdescriptor) {
+        /* Microdescriptors don't contain useful timestamps for this purpose,
+         * but we already knew that, so there's no need to log a warning
+         * further down below. */
+        continue;
+      } else if (descriptor instanceof RelayDirectory) {
+        publishedMillis = ((RelayDirectory) descriptor)
+            .getPublishedMillis();
+      } else if (descriptor instanceof RelayNetworkStatus) {
+        publishedMillis = ((RelayNetworkStatus) descriptor)
+            .getPublishedMillis();
+      } else if (descriptor instanceof RelayNetworkStatusConsensus) {
+        publishedMillis = ((RelayNetworkStatusConsensus) descriptor)
+            .getValidAfterMillis();
+      } else if (descriptor instanceof RelayNetworkStatusVote) {
+        publishedMillis = ((RelayNetworkStatusVote) descriptor)
+            .getValidAfterMillis();
+      } else if (descriptor instanceof ServerDescriptor) {
+        publishedMillis = ((ServerDescriptor) descriptor)
+            .getPublishedMillis();
+      } else if (descriptor instanceof SnowflakeStats) {
+        publishedMillis = ((SnowflakeStats) descriptor)
+            .snowflakeStatsEnd().toInstant(ZoneOffset.UTC).toEpochMilli();
+      } else if (descriptor instanceof TorperfResult) {
+        publishedMillis = ((TorperfResult) descriptor)
+            .getStartMillis();
+      } else if (descriptor instanceof WebServerAccessLog) {
+        publishedMillis = ((WebServerAccessLog) descriptor)
+            .getLogDate().atStartOfDay(ZoneOffset.UTC)
+            .toInstant().toEpochMilli();
+      } else {
+        /* Skip published timestamp if descriptor type is unknown or doesn't
+         * contain such a timestamp. */
+        unknownDescriptorSubclasses.add(
+            descriptor.getClass().getSimpleName());
+        continue;
+      }
+      if (null == firstPublishedMillis
+          || publishedMillis < firstPublishedMillis) {
+        firstPublishedMillis = publishedMillis;
+      }
+      if (null == lastPublishedMillis
+          || publishedMillis > lastPublishedMillis) {
+        lastPublishedMillis = publishedMillis;
+      }
+    }
+    if (!unknownDescriptorSubclasses.isEmpty()) {
+      logger.warn("Ran into unknown/unexpected Descriptor subclass(es) in "
+          + "{}: {}. Ignoring for index.json, but maybe worth looking into.",
+          this.path, unknownDescriptorSubclasses);
+    }
+    this.indexResult.firstPublished = null == firstPublishedMillis ? null
+        : dateTimeFormatter.format(Instant.ofEpochMilli(firstPublishedMillis));
+    this.indexResult.lastPublished = null == lastPublishedMillis ? null
+        : dateTimeFormatter.format(Instant.ofEpochMilli(lastPublishedMillis));
+  }
+}
+
diff --git a/src/main/resources/collector.properties b/src/main/resources/collector.properties
index e7cadf7..65e0f99 100644
--- a/src/main/resources/collector.properties
+++ b/src/main/resources/collector.properties
@@ -76,16 +76,11 @@ BridgedbMetricsOffsetMinutes = 340
 # The URL of this instance.  This will be the base URL
 # written to index.json, i.e. please change this to the mirrors url!
 InstanceBaseUrl = https://collector.torproject.org
-# The target location for index.json and its compressed
-# versions index.json.gz, index.json.bz2, and index.json.xz
-IndexPath = index
 # The top-level directory for archived descriptors.
-ArchivePath = archive
-# The top-level directory for third party data.
-ContribPath = contrib
+IndexedPath = indexed
 # The top-level directory for the recent descriptors that were
 # published in the last 72 hours.
-RecentPath = recent
+RecentPath = indexed/recent
 # The top-level directory for the retrieved descriptors that will
 # be archived.
 OutputPath = out
@@ -93,6 +88,11 @@ OutputPath = out
 StatsPath = stats
 # Path for descriptors downloaded from other instances
 SyncPath = sync
+# Directory served via an external web server and managed by us which contains
+# (hard) links to files in ArchivePath and RecentPath and which therefore must
+# be located on the same file system. Also contains index.json and its
+# compressed versions index.json.gz, index.json.bz2, and index.json.xz.
+HtdocsPath = htdocs
 ######## Relay descriptors ########
 #
 ## Define descriptor sources
diff --git a/src/main/resources/create-tarballs.sh b/src/main/resources/create-tarballs.sh
index 5802020..695bb24 100755
--- a/src/main/resources/create-tarballs.sh
+++ b/src/main/resources/create-tarballs.sh
@@ -10,7 +10,7 @@
 # Configuration section:
 # The following path should be adjusted, if the CollecTor server layout differs.
 # All paths should be given absolute.
-ARCHIVEDIR="/srv/collector.torproject.org/collector/archive"
+ARCHIVEDIR="/srv/collector.torproject.org/collector/indexed/archive"
 WORKDIR="/srv/collector.torproject.org/collector/tarballs"
 OUTDIR="/srv/collector.torproject.org/collector/out"
 TARBALLTARGETDIR="/srv/collector.torproject.org/collector/data"
diff --git a/src/test/java/org/torproject/metrics/collector/conf/ConfigurationTest.java b/src/test/java/org/torproject/metrics/collector/conf/ConfigurationTest.java
index 7e9ea28..887f3ae 100644
--- a/src/test/java/org/torproject/metrics/collector/conf/ConfigurationTest.java
+++ b/src/test/java/org/torproject/metrics/collector/conf/ConfigurationTest.java
@@ -39,7 +39,7 @@ public class ConfigurationTest {
   public void testKeyCount() {
     assertEquals("The number of properties keys in enum Key changed."
         + "\n This test class should be adapted.",
-        71, Key.values().length);
+        70, Key.values().length);
   }
 
   @Test()
diff --git a/src/test/java/org/torproject/metrics/collector/indexer/CreateIndexJsonTest.java b/src/test/java/org/torproject/metrics/collector/indexer/CreateIndexJsonTest.java
new file mode 100644
index 0000000..db00032
--- /dev/null
+++ b/src/test/java/org/torproject/metrics/collector/indexer/CreateIndexJsonTest.java
@@ -0,0 +1,522 @@
+/* Copyright 2019 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.metrics.collector.indexer;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertFalse;
+import static org.junit.Assert.assertTrue;
+import static org.junit.Assert.fail;
+
+import org.torproject.metrics.collector.conf.Configuration;
+import org.torproject.metrics.collector.conf.Key;
+
+import org.junit.Before;
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.TemporaryFolder;
+
+import java.io.IOException;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.nio.file.Paths;
+import java.nio.file.attribute.FileTime;
+import java.time.Instant;
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.List;
+
+/**
+ * Test class for {@link CreateIndexJson}.
+ */
+public class CreateIndexJsonTest {
+
+  /**
+   * Mocked indexer task that does not actually index a file but instead waits
+   * for the test class to set index results.
+   */
+  static class MockedIndexerTask extends IndexerTask {
+
+    /**
+     * Index result, to be set by the test.
+     */
+    private FileNode result;
+
+    /**
+     * Create a new mocked indexer task for the given path.
+     *
+     * @param path Path to index.
+     */
+    MockedIndexerTask(Path path) {
+      super(path);
+    }
+
+    /**
+     * Set index results.
+     *
+     * @param result Index results.
+     */
+    synchronized void setResult(FileNode result) {
+      this.result = result;
+      this.notifyAll();
+    }
+
+    /**
+     * Execute the task by waiting for the test to set index results.
+     *
+     * @return Index results provided by the test.
+     */
+    @Override
+    public FileNode call() {
+      synchronized (this) {
+        while (null == result) {
+          try {
+            wait();
+          } catch (InterruptedException e) {
+            /* Don't care about being interrupted, just keep waiting. */
+          }
+        }
+        return this.result;
+      }
+    }
+  }
+
+  /**
+   * List of mocked indexer tasks in the order of creation.
+   */
+  private List<MockedIndexerTask> indexerTasks = new ArrayList<>();
+
+  /**
+   * Testable version of the class under test.
+   */
+  class TestableCreateIndexJson extends CreateIndexJson {
+
+    /**
+     * Create a new instance with the given configuration.
+     *
+     * @param configuration Configuration for this test.
+     */
+    TestableCreateIndexJson(Configuration configuration) {
+      super(configuration);
+    }
+
+    /**
+     * Create an indexer task that doesn't actually index a file but that can
+     * be controlled by the test, and add that task to the list of tasks.
+     *
+     * @param fileToIndex File to index.
+     * @return Created (mocked) indexer task.
+     */
+    @Override
+    protected IndexerTask createIndexerTask(Path fileToIndex) {
+      MockedIndexerTask indexerTask = new MockedIndexerTask(fileToIndex);
+      indexerTasks.add(indexerTask);
+      return indexerTask;
+    }
+
+    /**
+     * Return {@code null} as build revision string to make it easier to compare
+     * written {@code index.json} files in tests.
+     *
+     * @return Always {@code null}.
+     */
+    @Override
+    protected String obtainBuildRevision() {
+      return null;
+    }
+  }
+
+  /**
+   * Temporary folder containing all files for this test.
+   */
+  @Rule
+  public TemporaryFolder temporaryFolder = new TemporaryFolder();
+
+  /**
+   * Path to recent exit list file in {@code indexed/recent/}.
+   */
+  private Path recentExitListFilePath;
+
+  /**
+   * Path to archived exit list file in {@code indexed/archive/}.
+   */
+  private Path archiveExitListFilePath;
+
+  /**
+   * Path to exit list link in {@code htdocs/recent/}.
+   */
+  private Path recentExitListLinkPath;
+
+  /**
+   * Path to {@code index.json} file in {@code htdocs/index/}.
+   */
+  private Path indexJsonPath;
+
+  /**
+   * Class under test.
+   */
+  private CreateIndexJson cij;
+
+  /**
+   * Prepares the temporary folder and configuration for this test.
+   *
+   * @throws IOException Thrown if an I/O error occurs.
+   */
+  @Before
+  public void prepareDirectoriesAndConfiguration() throws IOException {
+    Path indexedPath = this.temporaryFolder.newFolder("indexed").toPath();
+    this.recentExitListFilePath = indexedPath.resolve(
+        Paths.get("recent", "exit-lists", "2016-09-20-13-02-00"));
+    this.archiveExitListFilePath = indexedPath.resolve(
+        Paths.get("archive", "exit-lists", "exit-list-2016-09.tar.xz"));
+    Path htdocsPath = this.temporaryFolder.newFolder("htdocs").toPath();
+    this.recentExitListLinkPath = htdocsPath.resolve(
+        Paths.get("recent", "exit-lists", "2016-09-20-13-02-00"));
+    this.indexJsonPath = htdocsPath.resolve(
+        Paths.get("index", "index.json"));
+    Configuration configuration = new Configuration();
+    configuration.setProperty(Key.IndexedPath.name(),
+        indexedPath.toAbsolutePath().toString());
+    configuration.setProperty(Key.HtdocsPath.name(),
+        htdocsPath.toAbsolutePath().toString());
+    configuration.setProperty(Key.InstanceBaseUrl.name(),
+        "https://collector.torproject.org");
+    this.cij = new TestableCreateIndexJson(configuration);
+  }
+
+  /**
+   * First execution time.
+   */
+  private static final Instant firstExecution
+      = Instant.parse("2016-09-20T13:04:00Z");
+
+  /**
+   * Second execution time, two minutes after the first execution time, which is
+   * the default rate for executing this module.
+   */
+  private static final Instant secondExecution
+      = Instant.parse("2016-09-20T13:06:00Z");
+
+  /**
+   * Third execution, three hours later than the second execution time, to see
+   * if links to files that have been marked for deletion are actually deleted.
+   */
+  private static final Instant thirdExecution
+      = Instant.parse("2016-09-20T16:06:00Z");
+
+  /**
+   * Index result from indexing recent exit list.
+   */
+  private FileNode recentExitListFileNode = FileNode.of(
+      "2016-09-20-13-02-00", 177_090L, "2016-09-20 13:02",
+      Collections.singletonList("tordnsel 1.0"), "2016-09-20 13:02",
+      "2016-09-20 13:02", "4aXdw+jQ5O33AS8n+fUOwD5ZzHCICnwzvxkK8fWDhdw=");
+
+  /**
+   * Index result from indexing archived exit list.
+   */
+  private FileNode archiveExitListFileNode = FileNode.of(
+      "exit-list-2016-09.tar.xz", 1_008_748L, "2016-10-04 03:31",
+      Collections.singletonList("tordnsel 1.0"), "2016-09-01 00:02",
+      "2016-09-30 23:02", "P4zUKVOJFtKzxOXpN3NLU0UBZTBqCAM95yDPJ5JH62g=");
+
+  /**
+   * Index result from indexing <i>updated</i> archived exit list.
+   */
+  private FileNode updatedArchiveExitListFileNode = FileNode.of(
+      "exit-list-2016-09.tar.xz", 1_008_748L, "2016-10-07 03:31",
+      Collections.singletonList("tordnsel 1.0"), "2016-09-01 00:02",
+      "2016-09-30 23:02", "P4zUKVOJFtKzxOXpN3NLU0UBZTBqCAM95yDPJ5JH62g=");
+
+  /**
+   * Finish the oldest indexer task by providing the given file node as index
+   * result.
+   *
+   * @param fileNode Index result.
+   */
+  private void finishIndexing(FileNode fileNode) {
+    assertFalse(this.indexerTasks.isEmpty());
+    this.indexerTasks.remove(0).setResult(fileNode);
+  }
+
+  /**
+   * (Almost) empty {@code index.json} file.
+   */
+  private static final String emptyIndexJsonString
+      = "{\"index_created\":\"2016-09-20 13:06\","
+      + "\"path\":\"https://collector.torproject.org\"}";
+
+  /**
+   * {@code index.json} file containing a single recent exit list.
+   */
+  private static final String recentExitListIndexJsonString
+      = "{\"index_created\":\"2016-09-20 13:06\","
+      + "\"path\":\"https://collector.torproject.org\",\"directories\":[{"
+      + "\"path\":\"recent\",\"directories\":[{"
+      + "\"path\":\"exit-lists\",\"files\":[{"
+      + "\"path\":\"2016-09-20-13-02-00\",\"size\":177090,"
+      + "\"last_modified\":\"2016-09-20 13:02\","
+      + "\"types\":[\"tordnsel 1.0\"],"
+      + "\"first_published\":\"2016-09-20 13:02\","
+      + "\"last_published\":\"2016-09-20 13:02\","
+      + "\"sha256\":\"4aXdw+jQ5O33AS8n+fUOwD5ZzHCICnwzvxkK8fWDhdw=\"}]}]}]}";
+
+  /**
+   * {@code index.json} file containing a single archived exit list with a
+   * placeholder for the last-modified time.
+   */
+  private static final String archiveExitListIndexJsonString
+      = "{\"index_created\":\"2016-09-20 13:06\","
+      + "\"path\":\"https://collector.torproject.org\",\"directories\":[{"
+      + "\"path\":\"archive\",\"directories\":[{"
+      + "\"path\":\"exit-lists\",\"files\":[{"
+      + "\"path\":\"exit-list-2016-09.tar.xz\",\"size\":1008748,"
+      + "\"last_modified\":\"%s\","
+      + "\"types\":[\"tordnsel 1.0\"],"
+      + "\"first_published\":\"2016-09-01 00:02\","
+      + "\"last_published\":\"2016-09-30 23:02\","
+      + "\"sha256\":\"P4zUKVOJFtKzxOXpN3NLU0UBZTBqCAM95yDPJ5JH62g=\"}]}]}]}";
+
+  /**
+   * Delete the given file.
+   *
+   * @param fileToDelete Path to file to delete.
+   */
+  private static void deleteFile(Path fileToDelete) {
+    try {
+      Files.delete(fileToDelete);
+    } catch (IOException e) {
+      fail(String.format("I/O error while deleting %s.", fileToDelete));
+    }
+  }
+
+  /**
+   * Create the given file.
+   *
+   * @param fileToCreate Path to file to create.
+   * @param lastModified Last-modified time of file to create.
+   */
+  private static void createFile(Path fileToCreate, Instant lastModified) {
+    try {
+      Files.createDirectories(fileToCreate.getParent());
+      Files.createFile(fileToCreate);
+      Files.setLastModifiedTime(fileToCreate, FileTime.from(lastModified));
+    } catch (IOException e) {
+      fail(String.format("I/O error while creating %s.", fileToCreate));
+    }
+  }
+
+  /**
+   * Return whether the given file exists.
+   *
+   * @param fileToCheck Path to file to check.
+   * @return Whether the file exists.
+   */
+  private boolean fileExists(Path fileToCheck) {
+    return Files.exists(fileToCheck);
+  }
+
+  /**
+   * Change last-modified time of the given file.
+   *
+   * @param fileToChange File to change.
+   * @param lastModified New last-modified time.
+   */
+  private void changeLastModified(Path fileToChange, Instant lastModified) {
+    try {
+      Files.setLastModifiedTime(fileToChange, FileTime.from(lastModified));
+    } catch (IOException e) {
+      fail(String.format("I/O error while changing last-modified time of %s.",
+          fileToChange));
+    }
+  }
+
+  /**
+   * Write the given string to the {@code index.json} file.
+   *
+   * @param indexJsonString String to write.
+   * @param formatArguments Optional format arguments.
+   */
+  private void writeIndexJson(String indexJsonString,
+      Object ... formatArguments) {
+    try {
+      Files.createDirectories(indexJsonPath.getParent());
+      Files.write(indexJsonPath,
+          String.format(indexJsonString, formatArguments).getBytes());
+    } catch (IOException e) {
+      fail("I/O error while writing index.json file.");
+    }
+  }
+
+  /**
+   * Read and return the first line from the {@code index.json} file.
+   *
+   * @return First line from the {@code index.json} file.
+   */
+  private String readIndexJson() {
+    try {
+      return Files.readAllLines(indexJsonPath).get(0);
+    } catch (IOException e) {
+      fail("I/O error while reading index.json file.");
+      return null;
+    }
+  }
+
+  /**
+   * Run the module with the given system time.
+   *
+   * @param now Time when running the module.
+   */
+  private void startProcessing(Instant now) {
+    this.cij.startProcessing(now);
+  }
+
+  /**
+   * Test whether two executions on an empty {@code indexed/} directory produce
+   * an {@code index.json} file without any files or directories.
+   */
+  @Test
+  public void testEmptyDirs() {
+    startProcessing(firstExecution);
+    startProcessing(secondExecution);
+    assertEquals(emptyIndexJsonString, readIndexJson());
+  }
+
+  /**
+   * Test whether a new exit list in {@code indexed/recent/} gets indexed and
+   * then included in {@code index.json}.
+   */
+  @Test
+  public void testNewRecentExitList() {
+    createFile(recentExitListFilePath, Instant.parse("2016-09-20T13:02:00Z"));
+    startProcessing(firstExecution);
+    finishIndexing(this.recentExitListFileNode);
+    startProcessing(secondExecution);
+    assertEquals(recentExitListIndexJsonString, readIndexJson());
+  }
+
+  /**
+   * Test whether an existing exit list in {@code indexed/recent/} that is
+   * already contained in {@code index.json} gets ignored by the indexers.
+   */
+  @Test
+  public void testExistingRecentExitList() {
+    createFile(recentExitListFilePath, Instant.parse("2016-09-20T13:02:00Z"));
+    writeIndexJson(recentExitListIndexJsonString);
+    startProcessing(firstExecution);
+    startProcessing(secondExecution);
+    assertEquals(recentExitListIndexJsonString, readIndexJson());
+  }
+
+  /**
+   * Test whether a deleted exit list in {@code indexed/recent/} is first
+   * removed from {@code index.json} and later deleted from
+   * {@code htdocs/recent/}.
+   */
+  @Test
+  public void testDeletedRecentExitList() {
+    createFile(recentExitListFilePath, Instant.parse("2016-09-20T13:02:00Z"));
+    writeIndexJson(recentExitListIndexJsonString);
+    startProcessing(firstExecution);
+    assertTrue(fileExists(recentExitListLinkPath));
+    deleteFile(recentExitListFilePath);
+    startProcessing(secondExecution);
+    assertEquals(emptyIndexJsonString, readIndexJson());
+    fileExists(recentExitListLinkPath);
+    assertTrue(fileExists(recentExitListLinkPath));
+    startProcessing(thirdExecution);
+    assertFalse(fileExists(recentExitListLinkPath));
+  }
+
+  /**
+   * Test whether a link in {@code htdocs/recent/} for which no corresponding
+   * file in {@code indexed/recent/} exists is eventually deleted.
+   */
+  @Test
+  public void testDeletedLink() {
+    createFile(recentExitListLinkPath, Instant.parse("2016-09-20T13:02:00Z"));
+    startProcessing(firstExecution);
+    assertTrue(Files.exists(recentExitListLinkPath));
+    startProcessing(secondExecution);
+    assertTrue(Files.exists(recentExitListLinkPath));
+    startProcessing(thirdExecution);
+    assertFalse(Files.exists(recentExitListLinkPath));
+  }
+
+  /**
+   * Test whether a tarball that gets deleted while being indexed is not
+   * included in {@code index.json} even after indexing is completed.
+   */
+  @Test
+  public void testIndexingDisappearingTarball() {
+    createFile(recentExitListFilePath, Instant.parse("2016-09-20T13:02:00Z"));
+    startProcessing(firstExecution);
+    deleteFile(recentExitListFilePath);
+    finishIndexing(recentExitListFileNode);
+    startProcessing(secondExecution);
+    assertEquals(emptyIndexJsonString, readIndexJson());
+  }
+
+  /**
+   * Test whether a tarball that gets updated in {@code indexed/archive/} gets
+   * re-indexed and updated in {@code index.json}.
+   */
+  @Test
+  public void testUpdatedFile() {
+    writeIndexJson(archiveExitListIndexJsonString, "2016-10-04 03:31");
+    createFile(archiveExitListFilePath, Instant.parse("2016-10-07T03:31:00Z"));
+    startProcessing(firstExecution);
+    finishIndexing(updatedArchiveExitListFileNode);
+    startProcessing(secondExecution);
+    assertEquals(String.format(archiveExitListIndexJsonString,
+        "2016-10-07 03:31"), readIndexJson());
+  }
+
+  /**
+   * Test whether a tarball that gets updated while being indexed is not
+   * included in {@code index.json} even after indexing is completed.
+   */
+  @Test
+  public void testUpdateFileWhileIndexing() {
+    createFile(archiveExitListFilePath, Instant.parse("2016-10-07T03:31:00Z"));
+    startProcessing(firstExecution);
+    changeLastModified(archiveExitListFilePath,
+        Instant.parse("2016-10-07T03:31:00Z"));
+    finishIndexing(archiveExitListFileNode);
+    startProcessing(secondExecution);
+    assertEquals(String.format(archiveExitListIndexJsonString,
+        "2016-10-04 03:31"), readIndexJson());
+  }
+
+  /**
+   * Test whether a tarball that gets updated after being indexed but before
+   * being included in {@code index.json} is not being updated in
+   * {@code index.json} until the updated file is being indexed. */
+  @Test
+  public void testUpdateFileAfterIndexing() {
+    createFile(archiveExitListFilePath, Instant.parse("2016-10-04T03:31:00Z"));
+    startProcessing(firstExecution);
+    finishIndexing(archiveExitListFileNode);
+    changeLastModified(archiveExitListFilePath,
+        Instant.parse("2016-10-07T03:31:00Z"));
+    startProcessing(secondExecution);
+    assertEquals(String.format(archiveExitListIndexJsonString,
+        "2016-10-04 03:31"), readIndexJson());
+  }
+
+  /**
+   * Test whether a long-running indexer task is being given the time to finish,
+   * rather than starting another task for the same file.
+   */
+  @Test
+  public void testLongRunningIndexerTask() {
+    createFile(archiveExitListFilePath, Instant.parse("2016-10-04T03:31:00Z"));
+    startProcessing(firstExecution);
+    startProcessing(secondExecution);
+    assertEquals(emptyIndexJsonString, readIndexJson());
+    finishIndexing(archiveExitListFileNode);
+    startProcessing(thirdExecution);
+    assertTrue(this.indexerTasks.isEmpty());
+  }
+}
+
diff --git a/src/test/java/org/torproject/metrics/collector/indexer/IndexerTaskTest.java b/src/test/java/org/torproject/metrics/collector/indexer/IndexerTaskTest.java
new file mode 100644
index 0000000..8e5e6f4
--- /dev/null
+++ b/src/test/java/org/torproject/metrics/collector/indexer/IndexerTaskTest.java
@@ -0,0 +1,226 @@
+/* Copyright 2019 The Tor Project
+ * See LICENSE for licensing information */
+
+package org.torproject.metrics.collector.indexer;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertTrue;
+import static org.junit.Assert.fail;
+
+import org.junit.Rule;
+import org.junit.Test;
+import org.junit.rules.TemporaryFolder;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+
+import java.io.IOException;
+import java.io.InputStream;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.nio.file.attribute.FileTime;
+import java.time.LocalDateTime;
+import java.time.ZoneOffset;
+import java.time.format.DateTimeFormatter;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.SortedSet;
+import java.util.TreeSet;
+
+/**
+ * Test class for {@link IndexerTask}.
+ */
+ at RunWith(Parameterized.class)
+public class IndexerTaskTest {
+
+  @Parameterized.Parameter
+  public String path;
+
+  @Parameterized.Parameter(1)
+  public Long size;
+
+  @Parameterized.Parameter(2)
+  public String lastModified;
+
+  @Parameterized.Parameter(3)
+  public String[] types;
+
+  @Parameterized.Parameter(4)
+  public String firstPublished;
+
+  @Parameterized.Parameter(5)
+  public String lastPublished;
+
+  @Parameterized.Parameter(6)
+  public String sha256;
+
+  /**
+   * Initialize test parameters.
+   *
+   * @return Test parameters.
+   */
+  @Parameterized.Parameters
+  public static Collection<Object[]> pathFilename() {
+    return Arrays.asList(new Object[][]{
+
+        {"2016-09-20-13-00-00-consensus", /* Path in src/test/resources/ */
+            1_618_103L, /* Size in bytes */
+            "2017-09-07 12:13", /* Last-modified time */
+            new String[] { "network-status-consensus-3 1.17" }, /* Types */
+            "2016-09-20 13:00", /* First published */
+            "2016-09-20 13:00", /* Last published */
+            "3mLpDZmP/NSgOgmuPDyljxh0Lup1L6FtD16266ZCGAw="}, /* SHA-256 */
+
+        {"2016-09-20-13-00-00-vote-49015F787433103580E3B66A1707A00E60F2D15B-"
+            + "60ADC6BEC262AE921A1037D54C8A3976367DBE87",
+            3_882_514L,
+            "2017-09-07 12:13",
+            new String[] { "network-status-vote-3 1.17" },
+            "2016-09-20 13:00",
+            "2016-09-20 13:00",
+            "UCnSSrvdm26dJOriFgEQNQVrBLpVKbH/fF0VPRX3TGc="},
+
+        {"2016-09-20-13-02-00",
+            177_090L,
+            "2017-01-13 16:55",
+            new String[] { "tordnsel 1.0" },
+            "2016-09-20 13:02",
+            "2016-09-20 13:02",
+            "4aXdw+jQ5O33AS8n+fUOwD5ZzHCICnwzvxkK8fWDhdw="},
+
+        {"2016-10-01-16-00-00-vote-0232AF901C31A04EE9848595AF9BB7620D4C5B2E-"
+            + "FEE63B4AB7CE5A6BDD09E9A5C4F01BD61EB7E4F1",
+            3_226_152L,
+            "2017-01-13 16:55",
+            new String[] { "network-status-vote-3 1.0" },
+            "2016-10-01 16:00",
+            "2016-10-01 16:00",
+            "bilv6zEXr0Y9f5o24RMN0lUujsJJiSQAn9LkG0XJrZE="},
+
+        {"2016-10-02-17-00-00-consensus-microdesc",
+            1_431_627L,
+            "2017-09-07 12:13",
+            new String[] { "network-status-microdesc-consensus-3 1.17" },
+            "2016-10-02 17:00",
+            "2016-10-02 17:00",
+            "rrkxuLahYENLExX99Jio587/kUz9NtOoaYyKXxvX5EA="},
+
+        {"20160920-063816-1D8F3A91C37C5D1C4C19B1AD1D0CFBE8BF72D8E1",
+            339_256L,
+            "2017-09-07 12:13",
+            new String[] { "bridge-network-status 1.17" },
+            "2016-09-20 06:38",
+            "2016-09-20 06:38",
+            "sMAcyFrZ2rxj50b6iGe3icCNMC4gBSA1y9ZH4EWTa8s="},
+
+        {"bridge-2016-10-02-08-09-00-extra-infos",
+            11_561L,
+            "2017-09-07 12:13",
+            new String[] { "bridge-extra-info 1.3" },
+            "2016-10-02 06:09",
+            "2016-10-02 06:09",
+            "hat+vbyE04eH9JBQa0s6ezB6sLaStUUhvUj8CZ1aoEY="},
+
+        {"bridge-2016-10-02-16-09-00-server-descriptors",
+            5_336L,
+            "2017-01-13 16:55",
+            new String[] { "bridge-server-descriptor 1.2" },
+            "2016-10-02 14:09",
+            "2016-10-02 14:09",
+            "6CtHdo+eRFOi5xBjJcOVszC1hibC5gTB+YWvn1VmIIc="},
+
+        {"moria-1048576-2016-10-05.tpf",
+            20_405L,
+            "2017-09-07 12:13",
+            new String[0],
+            null,
+            null,
+            "DZyk6c0lQQ7OVZo1cmA+SuxPA+1thmuiooVifQPPOiA="},
+
+        {"op-nl-1048576-2017-04-11.tpf",
+            4_220L,
+            "2017-09-20 12:14",
+            new String[] { "torperf 1.1" },
+            "2017-04-11 06:24",
+            "2017-04-11 15:54",
+            "Gwex5yN3+s2PrhekjA68XmPg+UorOfx7mUa4prd7Dt8="},
+
+        {"relay-2016-10-02-08-05-00-extra-infos",
+            20_541L,
+            "2017-01-13 16:55",
+            new String[] { "extra-info 1.0" },
+            "2016-10-02 07:01",
+            "2016-10-02 07:01",
+            "3ZSO3+9ed9OwMVPx2LcVIiJfC+O30eEXEdbz64Hrp0w="},
+
+        {"relay-2016-10-02-16-05-00-server-descriptors",
+            17_404L,
+            "2017-01-13 16:55",
+            new String[] { "server-descriptor 1.0" },
+            "2016-10-02 14:58",
+            "2016-10-02 15:01",
+            "uWKHHzq4+oVNdOGh0mfkLUSjwGrBlLtEtN2DtF5qcLU="},
+
+        {"siv-1048576-2016-10-03.tpf",
+            39_193L,
+            "2017-01-13 16:55",
+            new String[] { "torperf 1.0" },
+            "2016-10-03 00:02",
+            "2016-10-03 23:32",
+            "paaFPI6BVuIDQ32aIuHYNCuKmBvFxsDvVCCwp+oM0GE="},
+
+        {"torperf-51200-2016-10-02.tpf",
+            233_763L,
+            "2017-01-13 16:55",
+            new String[] { "torperf 1.0" },
+            "2016-10-02 00:00",
+            "2016-10-02 23:55",
+            "fqeVAXamvB4yQ/8UlZAxhJx0+1Y7IfipqIpOUqQ57rE="}
+    });
+  }
+
+  /**
+   * Formatter for all timestamps found in {@code index.json}.
+   */
+  private static DateTimeFormatter dateTimeFormatter = DateTimeFormatter
+      .ofPattern("uuuu-MM-dd HH:mm").withZone(ZoneOffset.UTC);
+
+  /**
+   * Temporary folder containing all files for this test.
+   */
+  @Rule
+  public TemporaryFolder temporaryFolder = new TemporaryFolder();
+
+  /**
+   * Test indexing a file.
+   *
+   * @throws IOException Thrown if an I/O error occurs.
+   */
+  @Test
+  public void testIndexFile() throws IOException {
+    Path indexedDirectory = this.temporaryFolder.newFolder().toPath();
+    Path temporaryFile = indexedDirectory.resolve(this.path);
+    try (InputStream is = getClass()
+        .getClassLoader().getResourceAsStream(this.path)) {
+      if (null == is) {
+        fail(String.format("Unable to read test resource %s.", this.path));
+        return;
+      }
+      Files.copy(is, temporaryFile);
+    }
+    Files.setLastModifiedTime(temporaryFile,
+        FileTime.from(LocalDateTime.parse(this.lastModified, dateTimeFormatter)
+            .toInstant(ZoneOffset.UTC)));
+    assertTrue(Files.exists(temporaryFile));
+    IndexerTask indexerTask = new IndexerTask(temporaryFile);
+    FileNode indexResult = indexerTask.call();
+    assertEquals(this.path, indexResult.path);
+    assertEquals(this.size, indexResult.size);
+    assertEquals(this.lastModified, indexResult.lastModified);
+    SortedSet<String> expectedTypes = new TreeSet<>(Arrays.asList(this.types));
+    assertEquals(expectedTypes, indexResult.types);
+    assertEquals(this.firstPublished, indexResult.firstPublished);
+    assertEquals(this.lastPublished, indexResult.lastPublished);
+    assertEquals(this.sha256, indexResult.sha256);
+  }
+}
+





More information about the tor-commits mailing list