[tor-commits] [metrics-lib/master] Implement a minimal relay descriptor reader.
karsten at torproject.org
karsten at torproject.org
Fri Dec 16 11:58:22 UTC 2011
commit f3e72d2f6581712c28cbd35541ffbf9bcb59dea6
Author: Karsten Loesing <karsten.loesing at gmx.net>
Date: Fri Dec 16 12:57:45 2011 +0100
Implement a minimal relay descriptor reader.
---
.../descriptor/impl/DescriptorFileImpl.java | 44 ++++++++
.../descriptor/impl/RelayDescriptorReaderImpl.java | 110 +++++++++++++++++++-
2 files changed, 150 insertions(+), 4 deletions(-)
diff --git a/src/org/torproject/descriptor/impl/DescriptorFileImpl.java b/src/org/torproject/descriptor/impl/DescriptorFileImpl.java
new file mode 100644
index 0000000..0dd984d
--- /dev/null
+++ b/src/org/torproject/descriptor/impl/DescriptorFileImpl.java
@@ -0,0 +1,44 @@
+/* Copyright 2011 The Tor Project
+ * See LICENSE for licensing information */
+package org.torproject.descriptor.impl;
+
+import java.io.File;
+import java.util.List;
+import org.torproject.descriptor.Descriptor;
+import org.torproject.descriptor.DescriptorFile;
+
+public class DescriptorFileImpl implements DescriptorFile {
+
+ private File directory;
+ protected void setDirectory(File directory) {
+ this.directory = directory;
+ }
+ public File getDirectory() {
+ return this.file;
+ }
+
+ private File file;
+ protected void setFile(File file) {
+ this.file = file;
+ }
+ public File getFile() {
+ return this.file;
+ }
+
+ private long lastModified;
+ protected void setLastModified(long lastModified) {
+ this.lastModified = lastModified;
+ }
+ public long getLastModified() {
+ return this.lastModified;
+ }
+
+ private List<Descriptor> descriptors;
+ protected void setDescriptors(List<Descriptor> descriptors) {
+ this.descriptors = descriptors;
+ }
+ public List<Descriptor> getDescriptors() {
+ return this.descriptors;
+ }
+}
+
diff --git a/src/org/torproject/descriptor/impl/RelayDescriptorReaderImpl.java b/src/org/torproject/descriptor/impl/RelayDescriptorReaderImpl.java
index 8421f80..064983e 100644
--- a/src/org/torproject/descriptor/impl/RelayDescriptorReaderImpl.java
+++ b/src/org/torproject/descriptor/impl/RelayDescriptorReaderImpl.java
@@ -2,38 +2,140 @@
* See LICENSE for licensing information */
package org.torproject.descriptor.impl;
+import java.io.BufferedInputStream;
+import java.io.ByteArrayOutputStream;
import java.io.File;
+import java.io.FileInputStream;
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.Arrays;
import java.util.Iterator;
+import java.util.List;
import java.util.Map;
import java.util.Set;
+import java.util.Stack;
+import org.torproject.descriptor.Descriptor;
import org.torproject.descriptor.DescriptorFile;
import org.torproject.descriptor.RelayDescriptorReader;
+import org.torproject.descriptor.RelayNetworkStatusConsensus;
+import org.torproject.descriptor.RelayNetworkStatusVote;
public class RelayDescriptorReaderImpl implements RelayDescriptorReader {
+ private List<File> directories = new ArrayList<File>();
public void addDirectory(File directory) {
- /* TODO Implement me. */
+ this.directories.add(directory);
}
public void setExcludeFile(File fileToExclude) {
+ throw new UnsupportedOperationException("Not implemented yet.");
/* TODO Implement me. */
}
public void setExcludeFiles(Set<File> filesToExclude) {
+ throw new UnsupportedOperationException("Not implemented yet.");
/* TODO Implement me. */
}
- public void setExcludeFile(File fileToExclude, long lastModifiedMillis) {
+ public void setExcludeFile(File fileToExclude,
+ long lastModifiedMillis) {
+ throw new UnsupportedOperationException("Not implemented yet.");
/* TODO Implement me. */
}
public void setExcludeFiles(Map<File, Long> filesToExclude) {
+ throw new UnsupportedOperationException("Not implemented yet.");
/* TODO Implement me. */
}
public Iterator<DescriptorFile> readDescriptors() {
- /* TODO Implement me. */
- return new BlockingIteratorImpl<DescriptorFile>();
+ BlockingIteratorImpl<DescriptorFile> descriptorQueue =
+ new BlockingIteratorImpl<DescriptorFile>();
+ DescriptorReader reader = new DescriptorReader(this.directories,
+ descriptorQueue);
+ new Thread(reader).start();
+ return descriptorQueue;
+ }
+
+ private static class DescriptorReader implements Runnable {
+ private List<File> directories;
+ private BlockingIteratorImpl<DescriptorFile> descriptorQueue;
+ private DescriptorReader(List<File> directories,
+ BlockingIteratorImpl<DescriptorFile> descriptorQueue) {
+ this.directories = directories;
+ this.descriptorQueue = descriptorQueue;
+ }
+ public void run() {
+ for (File directory : this.directories) {
+ try {
+ Stack<File> files = new Stack<File>();
+ files.add(directory);
+ while (!files.isEmpty()) {
+ File file = files.pop();
+ if (file.isDirectory()) {
+ files.addAll(Arrays.asList(file.listFiles()));
+ } else {
+ try {
+ List<Descriptor> parsedDescriptors = this.readFile(file);
+ DescriptorFileImpl descriptorFile =
+ new DescriptorFileImpl();
+ descriptorFile.setDirectory(directory);
+ descriptorFile.setFile(file);
+ descriptorFile.setLastModified(file.lastModified());
+ descriptorFile.setDescriptors(parsedDescriptors);
+ this.descriptorQueue.add(descriptorFile);
+ } catch (DescriptorParseException e) {
+ /* TODO Handle me. */
+ }
+ }
+ }
+ } catch (IOException e) {
+ System.err.println("Error while reading descriptors in '"
+ + directory.getAbsolutePath() + "'.");
+ /* TODO Handle this exception somehow. */
+ } finally {
+ this.descriptorQueue.setOutOfDescriptors();
+ }
+ }
+ }
+ private List<Descriptor> readFile(File file) throws IOException,
+ DescriptorParseException {
+ FileInputStream fis = new FileInputStream(file);
+ BufferedInputStream bis = new BufferedInputStream(fis);
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ int len;
+ byte[] data = new byte[1024];
+ while ((len = bis.read(data, 0, 1024)) >= 0) {
+ baos.write(data, 0, len);
+ }
+ bis.close();
+ byte[] rawDescriptorBytes = baos.toByteArray();
+ /* TODO This consensus/vote detection is a hack. */
+ boolean isConsensus = false;
+ if (file.getName().contains("consensus")) {
+ isConsensus = true;
+ } else if (file.getName().contains("vote")) {
+ } else {
+ throw new RuntimeException("Unknown descriptor type in '"
+ + file.getAbsolutePath() + ".");
+ }
+ return this.parseDescriptors(rawDescriptorBytes, isConsensus);
+ }
+ private List<Descriptor> parseDescriptors(byte[] rawDescriptorBytes,
+ boolean isConsensus) throws DescriptorParseException {
+ List<Descriptor> parsedDescriptors = new ArrayList<Descriptor>();
+ if (isConsensus) {
+ List<RelayNetworkStatusConsensus> parsedConsensuses =
+ RelayNetworkStatusConsensusImpl.parseConsensuses(
+ rawDescriptorBytes);
+ parsedDescriptors.addAll(parsedConsensuses);
+ } else {
+ List<RelayNetworkStatusVote> parsedVotes =
+ RelayNetworkStatusVoteImpl.parseVotes(rawDescriptorBytes);
+ parsedDescriptors.addAll(parsedVotes);
+ }
+ return parsedDescriptors;
+ }
}
}
More information about the tor-commits
mailing list