[Author Prev][Author Next][Thread Prev][Thread Next][Author Index][Thread Index]
[or-cvs] [metrics-utils/master 2/2] Check in ExoneraTor and bridge descriptor sanitizer from SVN.
Author: Karsten Loesing <karsten.loesing@xxxxxxx>
Date: Mon, 20 Sep 2010 10:31:57 +0200
Subject: Check in ExoneraTor and bridge descriptor sanitizer from SVN.
Commit: d2c24d70f7bd92eeeeac71c48b80fe6e3b9b2fca
---
bridge-desc-sanitizer/ConvertBridgeDescs.java | 452 +++++++++++++++++++++++++
bridge-desc-sanitizer/HOWTO | 113 ++++++
bridge-desc-sanitizer/extract-bridges.sh | 8 +
exonerator/ExoneraTor.java | 404 ++++++++++++++++++++++
exonerator/HOWTO | 159 +++++++++
exonerator/LICENSE | 30 ++
exonerator/exonerator.py | 371 ++++++++++++++++++++
7 files changed, 1537 insertions(+), 0 deletions(-)
create mode 100644 bridge-desc-sanitizer/ConvertBridgeDescs.java
create mode 100644 bridge-desc-sanitizer/HOWTO
create mode 100755 bridge-desc-sanitizer/extract-bridges.sh
create mode 100644 exonerator/ExoneraTor.java
create mode 100644 exonerator/HOWTO
create mode 100644 exonerator/LICENSE
create mode 100755 exonerator/exonerator.py
diff --git a/bridge-desc-sanitizer/ConvertBridgeDescs.java b/bridge-desc-sanitizer/ConvertBridgeDescs.java
new file mode 100644
index 0000000..6a6c5bf
--- /dev/null
+++ b/bridge-desc-sanitizer/ConvertBridgeDescs.java
@@ -0,0 +1,452 @@
+import java.io.*;
+import java.util.*;
+import org.apache.commons.codec.digest.DigestUtils;
+import org.apache.commons.codec.binary.*;
+
+public class ConvertBridgeDescs {
+
+ public static void main(String[] args) throws Exception {
+
+ long started = System.currentTimeMillis();
+
+ if (args.length < 5) {
+ System.err.println("Usage: java "
+ + ConvertBridgeDescs.class.getSimpleName()
+ + " <input directory> <geoip.txt file> <YYYY> <MM> "
+ + "<output directory>");
+ System.exit(1);
+ }
+ File inDir = new File(args[0]);
+ File geoipFile = new File(args[1]);
+ String year = args[2];
+ String month = args[3];
+ int yearInt = Integer.parseInt(year);
+ int monthInt = Integer.parseInt(month);
+ File outDir = new File(args[4]);
+ if (!outDir.exists()) {
+ outDir.mkdir();
+ }
+
+ SortedSet<File> statuses = new TreeSet<File>();
+ Set<File> descriptors = new HashSet<File>();
+ Set<File> extrainfos = new HashSet<File>();
+
+ System.out.print("Parsing geoip.txt file... ");
+ BufferedReader r = new BufferedReader(new FileReader(geoipFile));
+ String line0 = null;
+ SortedMap<Long, String> geoipDatabase = new TreeMap<Long, String>();
+ while ((line0 = r.readLine()) != null) {
+ if (!line0.startsWith("#"))
+ geoipDatabase.put(Long.parseLong(line0.split(",")[0]),
+ line0.substring(line0.indexOf(',') + 1));
+ }
+ System.out.println("Found " + geoipDatabase.size()
+ + " entries (expected 100,000 +- 10,000).");
+
+ System.out.println("Checking files in " + inDir.getAbsolutePath()
+ + "...");
+ Stack<File> directoriesLeftToParse = new Stack<File>();
+ directoriesLeftToParse.push(inDir);
+ String currentYearAndMonth = "from-tonga-" + year + "-" + month;
+ String previousYearAndMonth = "from-tonga-" + (monthInt == 1 ?
+ "" + (yearInt - 1) + "-12" :
+ year + "-" + (monthInt < 11 ? "0" : "") + (monthInt - 1));
+ String nextYearAndMonth = "from-tonga-" + (monthInt == 12 ?
+ "" + (yearInt + 1) + "-01" :
+ year + "-" + (monthInt < 9 ? "0" : "") + (monthInt + 1));
+ while (!directoriesLeftToParse.isEmpty()) {
+ File directoryOrFile = directoriesLeftToParse.pop();
+ String filename = directoryOrFile.getName();
+ boolean addDirectory = false;
+ if (directoryOrFile.isDirectory()) {
+ if (/* base directory */
+ filename.equals("in") ||
+ /* current month */
+ filename.startsWith(currentYearAndMonth) ||
+ /* last days of previous month */
+ (filename.startsWith(previousYearAndMonth)
+ && Integer.parseInt(filename.substring(19, 21)) > 24) ||
+ /* first days of next month */
+ (filename.startsWith(nextYearAndMonth)
+ && Integer.parseInt(filename.substring(19, 21)) < 6)) {
+ for (File fileInDir: directoryOrFile.listFiles()) {
+ directoriesLeftToParse.push(fileInDir);
+ }
+ }
+ continue;
+ }
+ if (filename.startsWith("cached-extrainfo")) {
+ extrainfos.add(directoryOrFile);
+ } else if (filename.equals("bridge-descriptors")) {
+ descriptors.add(directoryOrFile);
+ } else if (filename.equals("networkstatus-bridges")) {
+ statuses.add(directoryOrFile);
+ }
+ }
+
+ int days = ((extrainfos.size() / 2 + descriptors.size()
+ + statuses.size()) + 3 * 24) / (3 * 48);
+ System.out.println("Found " + extrainfos.size()
+ + " cached-extrainfo[.new] files, " + descriptors.size()
+ + " bridge-descriptors files, and " + statuses.size()
+ + " networkstatus-bridges files, covering approximately " + days
+ + " days.");
+
+ System.out.print("Parsing extra-info descriptors");
+ String[] hex = new String[] { "0", "1", "2", "3", "4", "5", "6", "7",
+ "8", "9", "a", "b", "c", "d", "e", "f" };
+ for (String x : hex)
+ for (String y : hex)
+ new File(outDir + File.separator + "extra-infos" + File.separator
+ + x + File.separator + y).mkdirs();
+ Set<File> writtenExtrainfos = new HashSet<File>();
+ Map<String, String> extrainfoMapping = new HashMap<String, String>();
+ int parsed = 0;
+ for (File file : extrainfos) {
+ if (parsed++ > extrainfos.size() / days) {
+ System.out.print(".");
+ parsed = 0;
+ }
+ BufferedReader br = new BufferedReader(new FileReader(file));
+ String line = null;
+ StringBuilder original = null, scrubbed = null;
+ boolean skipSignature = false;
+ while ((line = br.readLine()) != null) {
+ if (skipSignature && !line.equals("-----END SIGNATURE-----")) {
+ continue;
+ } else if (line.startsWith("extra-info ")) {
+ original = new StringBuilder(line + "\n");
+ scrubbed = new StringBuilder("extra-info Unnamed "
+ + DigestUtils.shaHex(Hex.decodeHex(
+ line.split(" ")[2].toCharArray())).toUpperCase() + "\n");
+ } else if (line.startsWith("published ")
+ || line.startsWith("write-history ")
+ || line.startsWith("read-history ")
+ || line.startsWith("geoip-start-time ")
+ || line.startsWith("geoip-client-origins ")) {
+ original.append(line + "\n");
+ scrubbed.append(line + "\n");
+ } else if (line.startsWith("router-signature")) {
+ String originalDesc = original.toString() + line + "\n";
+ String originalHash = DigestUtils.shaHex(originalDesc);
+ String scrubbedDesc = scrubbed.toString();
+ String scrubbedHash = DigestUtils.shaHex(scrubbedDesc);
+ if (extrainfoMapping.containsKey(originalHash) &&
+ !extrainfoMapping.get(originalHash).equals(scrubbedHash)) {
+ System.out.println("We already have an extra-info mapping "
+ + "from " + originalHash + " to "
+ + extrainfoMapping.get(originalHash) + ", but we now want "
+ + "to add a mapping to " + scrubbedHash + ". Exiting");
+ System.exit(1);
+ }
+ extrainfoMapping.put(originalHash, scrubbedHash);
+ File out = new File(outDir + File.separator + "extra-infos"
+ + File.separator + scrubbedHash.charAt(0) + File.separator
+ + scrubbedHash.charAt(1) + File.separator + scrubbedHash);
+ if (!out.exists()) {
+ BufferedWriter bw = new BufferedWriter(new FileWriter(out));
+ bw.write(scrubbedDesc);
+ bw.close();
+ writtenExtrainfos.add(out);
+ }
+ } else if (line.equals("-----BEGIN SIGNATURE-----")) {
+ skipSignature = true;
+ } else if (line.equals("-----END SIGNATURE-----")) {
+ skipSignature = false;
+ } else {
+ System.out.println("Unrecognized line '" + line + "'. Exiting");
+ System.exit(1);
+ }
+ }
+ br.close();
+ }
+ System.out.println("\nWrote " + writtenExtrainfos.size()
+ + " extra-info descriptors.");
+
+ System.out.print("Parsing server descriptors");
+ for (String x : hex)
+ for (String y : hex)
+ new File(outDir + File.separator + "descriptors" + File.separator
+ + x + File.separator + y).mkdirs();
+ Set<File> writtenDescriptors = new HashSet<File>();
+ Map<File, File> referencedExtraInfos = new HashMap<File, File>();
+ Map<String, String> descriptorMapping = new HashMap<String, String>();
+ int found = 0, notfound = 0;
+ parsed = 0;
+ String haveExtraInfo = null;
+ for (File file : descriptors) {
+ if (parsed++ > descriptors.size() / days) {
+ System.out.print(".");
+ parsed = 0;
+ }
+ BufferedReader br = new BufferedReader(new FileReader(file));
+ String line = null, country = null;
+ StringBuilder original = null, scrubbed = null;
+ boolean skipCrypto = false, contactWritten = false;
+ while ((line = br.readLine()) != null) {
+ if (skipCrypto && !line.startsWith("-----END ")) {
+ original.append(line + "\n");
+ continue;
+ } else if (line.startsWith("router ")) {
+ original = new StringBuilder(line + "\n");
+ country = "zz";
+ String[] ipParts = line.split(" ")[2].replace('.', ' ').split(" ");
+ long ipNum = Long.parseLong(ipParts[0]) * 256L * 256L * 256L
+ + Long.parseLong(ipParts[1]) * 256L * 256L
+ + Long.parseLong(ipParts[2]) * 256L
+ + Long.parseLong(ipParts[3]);
+ long intervalStart = -1;
+ if (ipNum >= geoipDatabase.firstKey()) {
+ intervalStart = geoipDatabase.subMap(0L, ipNum).lastKey();
+ String dbContent = geoipDatabase.get(intervalStart);
+ long intervalEnd = Long.parseLong(dbContent.split(",")[0]);
+ if (ipNum <= intervalEnd)
+ country = dbContent.split(",")[1].toLowerCase();
+ }
+ scrubbed = new StringBuilder("router Unnamed 127.0.0.1 "
+ + line.split(" ")[3] + " " + line.split(" ")[4] + " "
+ + line.split(" ")[5] + "\n");
+ contactWritten = false;
+ haveExtraInfo = null;
+ } else if (line.startsWith("opt fingerprint ")) {
+ original.append(line + "\n");
+ scrubbed.append("opt fingerprint");
+ String fingerprint = DigestUtils.shaHex(Hex.decodeHex(
+ line.substring(16).replaceAll(" ", "").toCharArray())).
+ toUpperCase();
+ for (int i = 0; i < fingerprint.length() / 4; i++)
+ scrubbed.append(" " + fingerprint.substring(4 * i, 4 * (i + 1)));
+ scrubbed.append("\n");
+ } else if (line.startsWith("contact ")) {
+ original.append(line + "\n");
+ scrubbed.append("contact somebody at example dot " + country
+ + "\n");
+ contactWritten = true;
+ } else if (line.startsWith("router-signature")) {
+ String originalDesc = original.toString() + line + "\n";
+ String originalHash = DigestUtils.shaHex(originalDesc);
+ String scrubbedDesc = scrubbed.toString();
+ String scrubbedHash = DigestUtils.shaHex(scrubbedDesc);
+ if (descriptorMapping.containsKey(originalHash) &&
+ !descriptorMapping.get(originalHash).equals(scrubbedHash)) {
+ System.out.println("We already have a descriptor mapping "
+ + "from " + originalHash + " to "
+ + descriptorMapping.get(originalHash) + ", but we now "
+ + "want to add a mapping to " + scrubbedHash
+ + ". Exiting");
+ System.exit(1);
+ }
+ descriptorMapping.put(originalHash, scrubbedHash);
+ if (haveExtraInfo != null) {
+ File out = new File(outDir + File.separator + "descriptors"
+ + File.separator + scrubbedHash.charAt(0) + File.separator
+ + scrubbedHash.charAt(1) + File.separator + scrubbedHash);
+ if (!out.exists()) {
+ BufferedWriter bw2 = new BufferedWriter(new FileWriter(out));
+ bw2.write(scrubbedDesc);
+ bw2.close();
+ writtenDescriptors.add(out);
+ String extraInfoHash = haveExtraInfo.toLowerCase();
+ File extrainfoFile = new File(outDir + File.separator
+ + "extra-infos" + File.separator
+ + extraInfoHash.charAt(0) + File.separator
+ + extraInfoHash.charAt(1) + File.separator
+ + extraInfoHash);
+ if (!extrainfoFile.exists()) {
+ System.out.println("Extra-info descriptor '"
+ + extrainfoFile + "' does not exist.");
+ System.exit(1);
+ }
+ referencedExtraInfos.put(out, extrainfoFile);
+ }
+ }
+ } else if (line.startsWith("opt extra-info-digest ")) {
+ String originalExtraInfo = line.split(" ")[2].toLowerCase();
+ if (!extrainfoMapping.containsKey(originalExtraInfo)) {
+ notfound++;
+ } else {
+ found++;
+ original.append(line + "\n");
+ haveExtraInfo = extrainfoMapping.get(originalExtraInfo).
+ toUpperCase();
+ scrubbed.append("opt extra-info-digest " + haveExtraInfo
+ + "\n");
+ }
+ } else if (line.startsWith("reject ")
+ || line.startsWith("accept ")) {
+ if (!contactWritten) {
+ scrubbed.append("contact nobody at example dot " + country
+ + "\n");
+ contactWritten = true;
+ }
+ original.append(line + "\n");
+ scrubbed.append(line + "\n");
+ } else if (line.startsWith("platform ")
+ || line.startsWith("opt protocols ")
+ || line.startsWith("published ")
+ || line.startsWith("uptime ")
+ || line.startsWith("bandwidth ")
+ || line.startsWith("uptime ")
+ || line.startsWith("opt hibernating ")
+ || line.equals("opt hidden-service-dir")
+ || line.equals("opt caches-extra-info")) {
+ original.append(line + "\n");
+ scrubbed.append(line + "\n");
+ } else if (line.startsWith("family ")) {
+ StringBuilder familyLine = new StringBuilder("family");
+ for (String s : line.substring(7).split(" ")) {
+ if (s.startsWith("$"))
+ familyLine.append(" $" + DigestUtils.shaHex(Hex.decodeHex(
+ s.substring(1).toCharArray())).toUpperCase());
+ else
+ familyLine.append(" " + s);
+ }
+ original.append(line + "\n");
+ scrubbed.append(familyLine.toString() + "\n");
+ } else if (line.startsWith("@purpose ")) {
+ continue;
+ } else if (line.startsWith("-----BEGIN ")
+ || line.equals("onion-key") || line.equals("signing-key")) {
+ skipCrypto = true;
+ original.append(line + "\n");
+ } else if (line.startsWith("-----END ")) {
+ skipCrypto = false;
+ original.append(line + "\n");
+ } else {
+ System.out.println("Unrecognized line '" + line + "'. Exiting");
+ System.exit(1);
+ }
+ }
+ br.close();
+ }
+ System.out.println("\nWrote " + writtenDescriptors.size()
+ + " bridge descriptors. While parsing, we found that we parsed "
+ + found + " extra-info identifiers before, but are missing "
+ + notfound + ". (The number of missing identifiers should be "
+ + "significantly smaller.)");
+
+ System.out.print("Parsing network statuses");
+ Set<File> referencedDescriptors = new HashSet<File>();
+ parsed = notfound = found = 0;
+ for (File file : statuses) {
+ if (parsed++ > statuses.size() / days) {
+ System.out.print(".");
+ parsed = 0;
+ }
+ if (!file.getParent().substring(file.getParent().
+ indexOf("from-tonga-")).startsWith(currentYearAndMonth)) {
+ continue;
+ }
+ BufferedReader br = new BufferedReader(new FileReader(file));
+ String line = null;
+ StringBuilder scrubbed = new StringBuilder();
+ boolean addSLine = false;
+ while ((line = br.readLine()) != null) {
+ if (line.startsWith("r ")) {
+ String[] parts = line.split(" ");
+ String bridgeIdentity = parts[2] + "==";
+ String hexBridgeIdentity = Hex.encodeHexString(
+ Base64.decodeBase64(bridgeIdentity));
+ String hashedBridgeIdentity2 = Base64.encodeBase64String(
+ DigestUtils.sha(Base64.decodeBase64(bridgeIdentity))).
+ replace("=", "");
+ String hashedBridgeIdentity = Base64.encodeBase64String(
+ DigestUtils.sha(Base64.decodeBase64(bridgeIdentity))).
+ substring(0, 27);
+ String descIdentifier = parts[3] + "==";
+ String hexDescIdentifier = Hex.encodeHexString(
+ Base64.decodeBase64(descIdentifier));
+ if (!descriptorMapping.containsKey(hexDescIdentifier)) {
+ notfound++;
+ addSLine = false;
+ } else {
+ found++;
+ String refDesc = descriptorMapping.get(hexDescIdentifier).
+ toLowerCase();
+ File descriptorFile = new File(outDir + File.separator
+ + "descriptors" + File.separator + refDesc.charAt(0)
+ + File.separator + refDesc.charAt(1) + File.separator
+ + refDesc);
+ if (!descriptorFile.exists()) {
+ System.out.println("Descriptor file '"
+ + descriptorFile.getAbsolutePath() + "' does not exist.");
+ }
+ String replacementDescIdentifier = Base64.encodeBase64String(
+ Hex.decodeHex(descriptorMapping.get(hexDescIdentifier).
+ toCharArray())).substring(0, 27);
+ scrubbed.append("r Unnamed " + hashedBridgeIdentity
+ + " " + replacementDescIdentifier + " " + parts[4] + " "
+ + parts[5] + " 127.0.0.1 " + parts[7] + " " + parts[8]
+ + "\n");
+ addSLine = true;
+ referencedDescriptors.add(descriptorFile);
+ }
+ } else if (line.startsWith("s ")) {
+ if (addSLine) {
+ scrubbed.append(line + "\n");
+ }
+ } else {
+ System.out.println("Unknown line: " + line);
+ System.exit(1);
+ }
+ }
+ String timeString = file.getParent().substring(file.getParent().
+ indexOf("from-tonga-") + 11);
+ String[] date = timeString.substring(0, 10).split("-");
+ String time = timeString.substring(11, 17);
+ File dir = new File(outDir + File.separator + "statuses"
+ + File.separator + date[0] + File.separator + date[1]
+ + File.separator + date[2] + File.separator);
+ dir.mkdirs();
+ File out = new File(dir.getAbsolutePath() + File.separator + date[0]
+ + date[1] + date[2] + "-" + time + "-"
+ + "4A0CCD2DDC7995083D73F5D667100C8A5831F16D");
+ if (!out.exists()) {
+ BufferedWriter bw3 = new BufferedWriter(new FileWriter(out));
+ bw3.write(scrubbed.toString());
+ bw3.close();
+ }
+ }
+ System.out.println("\nWhile parsing, we found that we parsed "
+ + found + " bridge descriptors before, but are missing "
+ + notfound + ". (The number of missing identifiers should be "
+ + "significantly smaller.)");
+
+ Set<File> deleteFromReferencedExtraInfos = new HashSet<File>();
+ for (File e : referencedExtraInfos.keySet()) {
+ if (!referencedDescriptors.contains(e)) {
+ deleteFromReferencedExtraInfos.add(e);
+ }
+ }
+ for (File e : deleteFromReferencedExtraInfos) {
+ referencedExtraInfos.remove(e);
+ }
+ SortedSet<File> deleteDescriptors = new TreeSet<File>();
+ for (File e : writtenDescriptors) {
+ if (!referencedDescriptors.contains(e)) {
+ deleteDescriptors.add(e);
+ }
+ }
+ SortedSet<File> deleteExtraInfos = new TreeSet<File>();
+ for (File e : writtenExtrainfos) {
+ if (!referencedExtraInfos.values().contains(e)) {
+ deleteExtraInfos.add(e);
+ }
+ }
+ System.out.println("Deleting " + deleteDescriptors.size()
+ + " unreferenced bridge descriptors and "
+ + deleteExtraInfos.size() + " extra-info descriptors (keeping "
+ + (writtenDescriptors.size() - deleteDescriptors.size())
+ + " bridge descriptors and " + (writtenExtrainfos.size()
+ - deleteExtraInfos.size()) + " extra-info descriptors).");
+ for (File e : deleteDescriptors)
+ e.delete();
+ for (File e : deleteExtraInfos)
+ e.delete();
+
+ long finished = System.currentTimeMillis();
+ System.out.println("Processing took " + ((finished - started) / 1000)
+ + " seconds.");
+ }
+}
+
diff --git a/bridge-desc-sanitizer/HOWTO b/bridge-desc-sanitizer/HOWTO
new file mode 100644
index 0000000..b84d5ce
--- /dev/null
+++ b/bridge-desc-sanitizer/HOWTO
@@ -0,0 +1,113 @@
+Bridge descriptor sanitizer
+
+---------------------------------------------------------------------------
+
+Introduction:
+
+The bridge authority Tonga keeps a list of bridges in order to serve bridge
+addresses and descriptors to its clients. Every half hour, Tonga copies a
+snapshot of the known bridge descriptors to moria where these descriptors
+are archived for later statistical analysis. As a guiding principle, the
+Tor project makes all data that it uses for statistical analysis available
+to the interested public, in order to maximize transparency towards the
+community. However, the bridge descriptors contain the IP addresses and
+other contact information of bridges that must not be made public, or the
+purpose of bridges as non-public entry points into the Tor network would be
+obsolete. This script takes the half-hourly snapshots as input, removes all
+possibly sensitive information from the descriptors, and puts out the
+sanitized bridge descriptors that are safe to be published.
+
+---------------------------------------------------------------------------
+
+Processing steps:
+
+The following steps are taken to remove all potentially sensitive
+information from the bridge descriptors while keeping them useful for
+statistical analysis.
+
+1. Replace the bridge identity with its SHA1 value
+
+ Clients can request a bridge's current descriptor by sending its
+ identity string to the bridge authority. This is a feature to make
+ bridges on dynamic IP addresses useful. Therefore, the original
+ identities (and anything that could be used to derive them) need to be
+ removed from the descriptors. The bridge identity is replaced with its
+ SHA1 hash value. The idea is to have a consistent replacement that
+ remains stable over months or even years (without keeping a secret for a
+ keyed hash function).
+
+2. Remove all cryptographic keys and signatures
+
+ It would be straightforward to learn about the bridge identity from the
+ bridge's public key. Replacing keys by newly generated ones seemed to be
+ unnecessary (and would involve keeping a state over months/years), so
+ that all cryptographic objects have simply been removed.
+
+3. Replace IP address with 127.0.0.1
+
+ Of course, the IP address needs to be removed, too. However, the IP
+ address is resolved to a country code first and the result written to
+ the contact line as "somebody at example dot de" for Germany, etc. The
+ ports are kept unchanged though.
+
+4. Replace contact information
+
+ If there is contact information in a descriptor, the contact line is
+ changed to "somebody at ...". If there is none, a contact line is added
+ saying "nobody at ..." in order to put in the country code.
+
+5. Replace nickname with Unnamed
+
+ The bridge nicknames might give hints on the location of the bridge if
+ chosen without care; e.g. a bridge nickname might be very similar to the
+ operators' relay nicknames which might be located on adjacent IP
+ addresses. All bridge nicknames are therefore replaced with the string
+ Unnamed.
+
+Note that these processing steps only prevent people from learning about
+new bridge locations. People who already know a bridge identity or location
+can easily learn more about this bridge from the sanitized descriptors.
+This is useful for statistical analysis, e.g. to filter out bridges that
+have been running as relays before.
+
+---------------------------------------------------------------------------
+
+Quick Start:
+
+The following steps are necessary to process the half-hourly snapshots as
+collected by moria:
+
+- Install Java 5 or higher.
+
+- Download Apache Commons Codec 1.4 or higher for Base 64 and hex encoding
+ from http://commons.apache.org/codec/ and place the .jar (in the
+ following assumed to be commons-codec-1.4.jar) in the same directory as
+ this HOWTO file.
+
+- Copy the half-hourly snapshots named from-tonga-YYYY-MM-DDThhmmssZ.tar.gz
+ in a directory called data/ in the same directory as this HOWTO file.
+
+- Run ./extract-bridges.sh to extract the half-hourly snapshots in data/
+ to separate directories in the newly created subdirectory in/ .
+
+- Copy the geoip.txt from the Tor sources (from /src/config/) to the same
+ directory as this HOWTO file.
+
+- Compile the Java class using
+
+ $ javac -cp commons-codec-1.4.jar ConvertBridgeDescs.java
+
+- Run the script, providing it with the parameters it needs:
+
+ java -cp .:commons-codec-1.4.jar ConvertBridgeDescs
+ <input directory> <geoip.txt file>
+ <YYYY> <MM> <output directory>
+
+ Note that YYYY and MM specify the month that shall be processed. The other
+ descriptors in the input directory are ignored.
+
+ A sample invocation might be:
+
+ $ java -cp .:commons-codec-1.4.jar ConvertBridgeDescs in/ geoip.txt
+ 2008 10 out/
+
diff --git a/bridge-desc-sanitizer/extract-bridges.sh b/bridge-desc-sanitizer/extract-bridges.sh
new file mode 100755
index 0000000..5f412c3
--- /dev/null
+++ b/bridge-desc-sanitizer/extract-bridges.sh
@@ -0,0 +1,8 @@
+#!/bin/bash
+mkdir "in/"
+for i in `ls data/ | cut -c 1-29`
+do
+mkdir "in/"$i
+tar -C "in/"$i -xf "data/"$i".tar.gz"
+done
+
diff --git a/exonerator/ExoneraTor.java b/exonerator/ExoneraTor.java
new file mode 100644
index 0000000..eba3cca
--- /dev/null
+++ b/exonerator/ExoneraTor.java
@@ -0,0 +1,404 @@
+/* Copyright 2009 The Tor Project
+ * See LICENSE for licensing information */
+
+import java.io.*;
+import java.math.*;
+import java.text.*;
+import java.util.*;
+import org.bouncycastle.util.encoders.Base64;
+
+public final class ExoneraTor {
+
+ public static void main(final String[] args) throws Exception {
+
+ // check parameters
+ if (args.length < 4 || args.length > 5) {
+ System.err.println("\nUsage: java "
+ + ExoneraTor.class.getSimpleName()
+ + " <descriptor archive directory> <IP address in question> "
+ + "<timestamp, in UTC, formatted as YYYY-MM-DD hh:mm:ss> "
+ + "[<target address>[:<target port>]]\n");
+ return;
+ }
+ File archiveDirectory = new File(args[0]);
+ if (!archiveDirectory.exists() || !archiveDirectory.isDirectory()) {
+ System.err.println("\nDescriptor archive directory + "
+ + archiveDirectory.getAbsolutePath()
+ + " does not exist or is not a directory.\n");
+ return;
+ }
+ String relayIP = args[1];
+ String timestampStr = args[2] + " " + args[3];
+ SimpleDateFormat timeFormat = new SimpleDateFormat(
+ "yyyy-MM-dd HH:mm:ss");
+ timeFormat.setTimeZone(TimeZone.getTimeZone("UTC"));
+ long timestamp = timeFormat.parse(timestampStr).getTime();
+ String target = null, targetIP = null, targetPort = null;
+ String[] targetIPParts = null;
+ if (args.length > 4) {
+ target = args[4];
+ if (target.contains(":")) {
+ targetIP = target.split(":")[0];
+ targetPort = target.split(":")[1];
+ } else {
+ targetIP = target;
+ }
+ targetIPParts = targetIP.replace(".", " ").split(" ");
+ }
+ String DELIMITER = "--------------------------------------------------"
+ + "-------------------------";
+ System.out.println("\nTrying to find out whether " + relayIP + " was "
+ + "running as a Tor relay at " + timestampStr
+ + (target != null ? " permitting exiting to " + target : "")
+ + "...\n\n" + DELIMITER);
+
+ // check that we have the required archives
+ long timestampTooOld = timestamp - 300 * 60 * 1000;
+ long timestampFrom = timestamp - 180 * 60 * 1000;
+ long timestampTooNew = timestamp + 120 * 60 * 1000;
+ Calendar calTooOld = Calendar.getInstance(TimeZone.getTimeZone("UTC"));
+ Calendar calFrom = Calendar.getInstance(TimeZone.getTimeZone("UTC"));
+ Calendar calTooNew = Calendar.getInstance(TimeZone.getTimeZone("UTC"));
+ calTooOld.setTimeInMillis(timestampTooOld);
+ calFrom.setTimeInMillis(timestampFrom);
+ calTooNew.setTimeInMillis(timestampTooNew);
+ System.out.printf("%nChecking that relevant archives between "
+ + "%tF %<tT and %tF %<tT are available...%n", calTooOld,
+ calTooNew);
+ SortedSet<String> requiredDirs = new TreeSet<String>();
+ requiredDirs.add(String.format("consensuses-%tY-%<tm", calTooOld));
+ requiredDirs.add(String.format("consensuses-%tY-%<tm", calTooNew));
+ if (target != null) {
+ requiredDirs.add(String.format("server-descriptors-%tY-%<tm",
+ calTooOld));
+ requiredDirs.add(String.format("server-descriptors-%tY-%<tm",
+ calTooNew));
+ }
+ SortedSet<File> consensusDirs = new TreeSet<File>();
+ SortedSet<File> descriptorsDirs = new TreeSet<File>();
+ Stack<File> directoriesLeftToParse = new Stack<File>();
+ directoriesLeftToParse.push(archiveDirectory);
+ while (!directoriesLeftToParse.isEmpty()) {
+ File directoryOrFile = directoriesLeftToParse.pop();
+ if (directoryOrFile.getName().startsWith("consensuses-")) {
+ if (requiredDirs.contains(directoryOrFile.getName())) {
+ requiredDirs.remove(directoryOrFile.getName());
+ consensusDirs.add(directoryOrFile);
+ }
+ } else if (directoryOrFile.getName().startsWith(
+ "server-descriptors-")) {
+ if (requiredDirs.contains(directoryOrFile.getName())) {
+ requiredDirs.remove(directoryOrFile.getName());
+ descriptorsDirs.add(directoryOrFile);
+ }
+ } else {
+ for (File fileInDir : directoryOrFile.listFiles())
+ if (fileInDir.isDirectory())
+ directoriesLeftToParse.push(fileInDir);
+ }
+ }
+ for (File dir : consensusDirs)
+ System.out.println(" " + dir.getAbsolutePath());
+ for (File dir : descriptorsDirs)
+ System.out.println(" " + dir.getAbsolutePath());
+ if (!requiredDirs.isEmpty()) {
+ System.out.println("\nWe are missing consensuses and/or server "
+ + "descriptors. Please download these archives and extract them "
+ + "to your data directory. Be sure NOT to rename the extracted "
+ + "directories or the contained files.");
+ for (String dir : requiredDirs)
+ System.out.println(" " + dir + ".tar.bz2");
+ return;
+ }
+
+ // look for consensus files
+ System.out.printf("%nLooking for relevant consensuses between "
+ + "%tF %<tT and %s...%n", calFrom, timestampStr);
+ SortedSet<File> tooOldConsensuses = new TreeSet<File>();
+ SortedSet<File> relevantConsensuses = new TreeSet<File>();
+ SortedSet<File> tooNewConsensuses = new TreeSet<File>();
+ directoriesLeftToParse.clear();
+ for (File consensusDir : consensusDirs)
+ directoriesLeftToParse.push(consensusDir);
+ SimpleDateFormat consensusTimeFormat = new SimpleDateFormat(
+ "yyyy-MM-dd-HH-mm-ss");
+ consensusTimeFormat.setTimeZone(TimeZone.getTimeZone("UTC"));
+ while (!directoriesLeftToParse.isEmpty()) {
+ File directoryOrFile = directoriesLeftToParse.pop();
+ if (directoryOrFile.isDirectory()) {
+ for (File fileInDir : directoryOrFile.listFiles()) {
+ directoriesLeftToParse.push(fileInDir);
+ }
+ continue;
+ } else {
+ String filename = directoryOrFile.getName();
+ if (filename.endsWith("consensus")) {
+ long consensusTime = consensusTimeFormat.parse(
+ filename.substring(0, 19)).getTime();
+ if (consensusTime >= timestampTooOld &&
+ consensusTime < timestampFrom)
+ tooOldConsensuses.add(directoryOrFile);
+ else if (consensusTime >= timestampFrom &&
+ consensusTime <= timestamp)
+ relevantConsensuses.add(directoryOrFile);
+ else if (consensusTime > timestamp &&
+ consensusTime <= timestampTooNew)
+ tooNewConsensuses.add(directoryOrFile);
+ }
+ }
+ }
+ SortedSet<File> allConsensuses = new TreeSet<File>();
+ allConsensuses.addAll(tooOldConsensuses);
+ allConsensuses.addAll(relevantConsensuses);
+ allConsensuses.addAll(tooNewConsensuses);
+ if (allConsensuses.isEmpty()) {
+ System.out.println(" None found!\n\n" + DELIMITER + "\n\nResult is "
+ + "INDECISIVE!\n\nWe cannot make any statement about IP address "
+ + relayIP + " being a relay at " + timestampStr + " or not! We "
+ + "did not find any relevant consensuses preceding the given "
+ + "time. This either means that you did not download and "
+ + "extract the consensus archives preceding the hours before "
+ + "the given time, or (in rare cases) that the directory "
+ + "archives are missing the hours before the timestamp. Please "
+ + "check that your directory archives contain consensus files "
+ + "of the interval 5:00 hours before and 2:00 hours after the "
+ + "time you are looking for.\n");
+ return;
+ }
+ for (File f : relevantConsensuses)
+ System.out.println(" " + f.getAbsolutePath());
+
+ // parse consensuses to find descriptors belonging to the IP address
+ System.out.println("\nLooking for descriptor identifiers referenced "
+ + "in \"r \" lines in these consensuses containing IP address "
+ + relayIP + "...");
+ SortedSet<File> positiveConsensusesNoTarget = new TreeSet<File>();
+ Set<String> addressesInSameNetwork = new HashSet<String>();
+ SortedMap<String, Set<File>> relevantDescriptors =
+ new TreeMap<String, Set<File>>();
+ for (File consensus : allConsensuses) {
+ if (relevantConsensuses.contains(consensus))
+ System.out.println(" " + consensus.getAbsolutePath());
+ BufferedReader br = new BufferedReader(new FileReader(consensus));
+ String line;
+ while ((line = br.readLine()) != null) {
+ if (!line.startsWith("r "))
+ continue;
+ String[] parts = line.split(" ");
+ String address = parts[6];
+ if (address.equals(relayIP)) {
+ byte[] result = Base64.decode(parts[3] + "==");
+ String hex = new BigInteger(1, Base64.decode(parts[3] +
+ "==")).toString(16).substring(0, 40);
+ if (!relevantDescriptors.containsKey(hex))
+ relevantDescriptors.put(hex, new HashSet<File>());
+ relevantDescriptors.get(hex).add(consensus);
+ positiveConsensusesNoTarget.add(consensus);
+ if (relevantConsensuses.contains(consensus))
+ System.out.println(" \"" + line + "\" references "
+ + "descriptor " + hex);
+ } else {
+ if (relayIP.startsWith(address.substring(0,
+ address.lastIndexOf(".")))) {
+ addressesInSameNetwork.add(address);
+ }
+ }
+ }
+ br.close();
+ }
+ if (relevantDescriptors.isEmpty()) {
+ System.out.printf(" None found!\n\n" + DELIMITER + "\n\nResult is "
+ + "NEGATIVE with moderate certainty!\n\nWe did not find IP "
+ + "address " + relayIP + " in any of the consensuses that were "
+ + "published between %tF %<tT and %tF %<tT.\n\nA possible "
+ + "reason for false negatives is that the relay is using a "
+ + "different IP address when generating a descriptor than for "
+ + "exiting to the Internet. We hope to provide better checks "
+ + "for this case in the future.", calTooOld, calTooNew);
+ if (!addressesInSameNetwork.isEmpty()) {
+ System.out.println("\n\nThe following other IP addresses of Tor "
+ + "relays were found in the mentioned consensus files that "
+ + "are in the same /24 network and that could be related to "
+ + "IP address " + relayIP + ":");
+ for (String s : addressesInSameNetwork) {
+ System.out.println(" " + s);
+ }
+ }
+ System.out.println();
+ return;
+ }
+
+ // parse router descriptors to check exit policies
+ SortedSet<File> positiveConsensuses = new TreeSet<File>();
+ Set<String> missingDescriptors = new HashSet<String>();
+ if (target != null) {
+ System.out.println("\nChecking if referenced descriptors permit "
+ + "exiting to " + target + "...");
+ Set<String> descriptors = relevantDescriptors.keySet();
+ missingDescriptors.addAll(relevantDescriptors.keySet());
+ directoriesLeftToParse.clear();
+ for (File descriptorsDir : descriptorsDirs)
+ directoriesLeftToParse.push(descriptorsDir);
+ while (!directoriesLeftToParse.isEmpty()) {
+ File directoryOrFile = directoriesLeftToParse.pop();
+ if (directoryOrFile.isDirectory()) {
+ for (File fileInDir : directoryOrFile.listFiles()) {
+ directoriesLeftToParse.push(fileInDir);
+ }
+ continue;
+ } else {
+ String filename = directoryOrFile.getName();
+ for (String descriptor : descriptors) {
+ if (filename.equals(descriptor)) {
+ missingDescriptors.remove(descriptor);
+ BufferedReader br = new BufferedReader(
+ new FileReader(directoryOrFile));
+ String line;
+ while ((line = br.readLine()) != null) {
+ if (line.startsWith("reject ") ||
+ line.startsWith("accept ")) {
+ boolean ruleAccept = line.split(" ")[0].equals("accept");
+ String ruleAddress = line.split(" ")[1].split(":")[0];
+ if (!ruleAddress.equals("*")) {
+ if (!ruleAddress.contains("/") &&
+ !ruleAddress.equals(targetIP))
+ continue; // IP address does not match
+ String[] ruleIPParts = ruleAddress.split("/")[0].
+ replace(".", " ").split(" ");
+ int ruleNetwork = Integer.parseInt(
+ ruleAddress.split("/")[1]);
+ for (int i = 0; i < 4; i++) {
+ if (ruleNetwork == 0) {
+ break;
+ } else if (ruleNetwork >= 8) {
+ if (ruleIPParts[i].equals(targetIPParts[i]))
+ ruleNetwork -= 8;
+ else
+ break;
+ } else {
+ int mask = 255 ^ 255 >>> ruleNetwork;
+ if ((Integer.parseInt(ruleIPParts[i]) & mask) ==
+ (Integer.parseInt(targetIPParts[i]) & mask))
+ ruleNetwork = 0;
+ break;
+ }
+ }
+ if (ruleNetwork > 0)
+ continue; // IP address does not match
+ }
+ String rulePort = line.split(" ")[1].split(":")[1];
+ if (targetPort == null && !ruleAccept &&
+ !rulePort.equals("*"))
+ continue; // with no port given, we only consider
+ // reject :* rules as matching
+ if (targetPort != null) {
+ if (!rulePort.equals("*") &&
+ !targetPort.equals(rulePort))
+ continue; // ports do not match
+ }
+ boolean relevantMatch = false;
+ for (File f : relevantDescriptors.get(descriptor))
+ if (relevantConsensuses.contains(f))
+ relevantMatch = true;
+ if (relevantMatch)
+ System.out.println(" "
+ + directoryOrFile.getAbsolutePath() + " "
+ + (ruleAccept ? "permits" : "does not permit")
+ + " exiting to " + target + " according to rule \""
+ + line + "\"");
+ if (ruleAccept)
+ positiveConsensuses.addAll(
+ relevantDescriptors.get(descriptor));
+ break;
+ }
+ }
+ br.close();
+ }
+ }
+ }
+ }
+ }
+
+ // print out result
+ Set<File> matches = (target != null) ? positiveConsensuses
+ : positiveConsensusesNoTarget;
+ if (matches.contains(relevantConsensuses.last())) {
+ System.out.println("\n" + DELIMITER + "\n\nResult is POSITIVE with "
+ + "high certainty!\n\nWe found one or more relays on IP address "
+ + relayIP
+ + (target != null ? " permitting exit to " + target : "")
+ + " in the most recent consensus preceding " + timestampStr
+ + " that clients were likely to know.\n");
+ return;
+ }
+ boolean resultIndecisive = target != null
+ && !missingDescriptors.isEmpty();
+ if (resultIndecisive) {
+ System.out.println("\n" + DELIMITER + "\n\nResult is INDECISIVE!\n\n"
+ + "At least one referenced descriptor could not be found. This "
+ + "is a rare case, but one that (apparently) happens. We cannot "
+ + "make any good statement about exit relays without these "
+ + "descriptors. The following descriptors are missing:");
+ for (String desc : missingDescriptors)
+ System.out.println(" " + desc);
+ }
+ boolean inOtherRelevantConsensus = false, inTooOldConsensuses = false,
+ inTooNewConsensuses = false;
+ for (File f : matches)
+ if (relevantConsensuses.contains(f))
+ inOtherRelevantConsensus = true;
+ else if (tooOldConsensuses.contains(f))
+ inTooOldConsensuses = true;
+ else if (tooNewConsensuses.contains(f))
+ inTooNewConsensuses = true;
+ if (inOtherRelevantConsensus) {
+ if (!resultIndecisive)
+ System.out.println("\n" + DELIMITER + "\n\nResult is POSITIVE "
+ + "with moderate certainty!");
+ System.out.println("\nWe found one or more relays on IP address "
+ + relayIP
+ + (target != null ? " permitting exit to " + target : "")
+ + ", but not in the consensus immediately preceding "
+ + timestampStr + ". A possible reason for the relay being "
+ + "missing in the last consensus preceding the given time might "
+ + "be that some of the directory authorities had difficulties "
+ + "connecting to the relay. However, clients might still have "
+ + "used the relay.");
+ } else {
+ if (!resultIndecisive)
+ System.out.println("\n" + DELIMITER + "\n\nResult is NEGATIVE "
+ + "with high certainty!");
+ System.out.println("\nWe did not find any relay on IP address "
+ + relayIP
+ + (target != null ? " permitting exit to " + target : "")
+ + " in the consensuses 3:00 hours preceding " + timestampStr
+ + ".");
+ if (inTooOldConsensuses || inTooNewConsensuses) {
+ if (inTooOldConsensuses && !inTooNewConsensuses)
+ System.out.println("\nNote that we found a matching relay in "
+ + "consensuses that were published between 5:00 and 3:00 "
+ + "hours before " + timestampStr + ".");
+ else if (!inTooOldConsensuses && inTooNewConsensuses)
+ System.out.println("\nNote that we found a matching relay in "
+ + "consensuses that were published up to 2:00 hours after "
+ + timestampStr + ".");
+ else
+ System.out.println("\nNote that we found a matching relay in "
+ + "consensuses that were published between 5:00 and 3:00 "
+ + "hours before and in consensuses that were published up "
+ + "to 2:00 hours after " + timestampStr + ".");
+ System.out.println("Make sure that the timestamp you provided is "
+ + "in the correct timezone: UTC (or GMT).");
+ }
+ }
+ if (target != null) {
+ if (positiveConsensuses.isEmpty() &&
+ !positiveConsensusesNoTarget.isEmpty())
+ System.out.println("\nNote that although the found relay(s) did "
+ + "not permit exiting to " + target + ", there have been one "
+ + "or more relays running at the given time.");
+ }
+ System.out.println();
+ }
+}
+
diff --git a/exonerator/HOWTO b/exonerator/HOWTO
new file mode 100644
index 0000000..907a8f5
--- /dev/null
+++ b/exonerator/HOWTO
@@ -0,0 +1,159 @@
+ExoneraTor
+ or: a script that tells you whether some IP address was a Tor relay
+
+---------------------------------------------------------------------------
+
+Introduction:
+
+Some people have expressed the desire to learn whether a given IP address
+has been a Tor relay at a certain time. In addition to that, these people
+might want to know whether the IP address permitted exit to a given address
+and port.
+
+Answering these questions can be important for Tor relay operators to show
+to the authorities that an anonymous user might have conducted bad things
+with their IP address. Likewise, police investigators might be interested
+in the answer to these questions, too, in order to decide whether to
+proceed with their investigations or not.
+
+We can answer the above questions from looking at the descriptor archives
+that are available since late 2007 (or even beyond, but this script only
+works with the data format that was produced starting in October 2007).
+This script parses the directory archives to print out the answer whether
+a certain IP address was a Tor relay at a given time. The script further
+prints out all intermediate steps in answering this, so that users can
+confirm the correctness of the result themselves.
+
+This script is available in two versions written in Python and in Java with
+equivalent functionality.
+
+---------------------------------------------------------------------------
+
+Python Quick Start:
+
+In order to run the Python version of this script, you need to install and
+download the following software and data (please note that all instructions
+are written for Linux; commands for Windows or Mac OS X may vary):
+
+- Install Python 2.6.2 or higher. (Previous Python versions might work,
+ too, but have not been tested.)
+
+- Install the Python module IPy 0.62 or higher either from
+ http://pypi.python.org/pypi/IPy/ or using "apt-get install python-ipy" on
+ Debian-based systems.
+
+- Download the v3 consensuses and server descriptors of the relevant time
+ from http://metrics.torproject.org/data.html and extract them to a
+ directory in your working directory, e.g. /home/you/exonerator/data/ .
+ Don't rename the extracted directories or any of the contained files, or
+ the script won't find the contained descriptors.
+
+ Note that you only need the server descriptors if you want to learn
+ whether a given IP address permits exiting to a given target. If you
+ only want to learn whether that IP address was a Tor relay, you don't
+ need them.
+
+- Run the script, providing it with the parameters it needs:
+
+ python exonerator.py [--archive=<descriptor archive directory>]
+ <IP address in question>
+ <timestamp, in UTC, formatted as YYYY-MM-DD hh:mm:ss>
+ [<target address>[:<target port>]]
+
+ The --archive option defaults to data/ . In the following examples, it is
+ assumed that this default applies.
+
+ Make sure that the timestamp is provided in UTC, which is equivalent to
+ GMT, and not in your local timezone! Otherwise, results will very likely
+ be wrong.
+
+ A sample invocation might be:
+
+ $ python exonerator.py 209.17.171.104 2009-08-15 16:05:00
+ 209.85.129.104:80
+
+---------------------------------------------------------------------------
+
+Java Quick Start:
+
+In order to run the Java version of this script, you need to install and
+download the following software and data (please note that all instructions
+are written for Linux; commands for Windows or Mac OS X may vary):
+
+- Install Java 6 or higher.
+
+- Download the BouncyCastle provider that includes Base 64 decoding from
+ http://www.bouncycastle.org/download/bcprov-jdk16-143.jar and put it in
+ your working directory, e.g. /home/you/exonerator/ .
+
+- Download the v3 consensuses and server descriptors of the relevant time
+ from http://metrics.torproject.org/data.html and extract them to a
+ directory in your working directory, e.g. /home/you/exonerator/data/ .
+ Don't rename the extracted directories or any of the contained files, or
+ the script won't find the contained descriptors.
+
+ Note that you only need the server descriptors if you want to learn
+ whether a given IP address permits exiting to a given target. If you
+ only want to learn whether that IP address was a Tor relay, you don't
+ need them.
+
+- Compile the (single) Java class using this command:
+
+ $ javac -cp bcprov-jdk16-143.jar ExoneraTor.java
+
+- Run the script, providing it with the parameters it needs:
+
+ java -cp .:bcprov-jdk16-143.jar ExoneraTor
+ <descriptor archive directory>
+ <IP address in question>
+ <timestamp, in UTC, formatted as YYYY-MM-DD hh:mm:ss>
+ [<target address>[:<target port>]]
+
+ Make sure that the timestamp is provided in UTC, which is equivalent to
+ GMT, and not in your local timezone! Otherwise, results will very likely
+ be wrong.
+
+ A sample invocation might be:
+
+ $ java -cp .:bcprov-jdk16-143.jar ExoneraTor data/ 209.17.171.104 \
+ 2009-08-15 16:05:00 209.85.129.104:80
+
+---------------------------------------------------------------------------
+
+Test cases:
+
+The following test cases work with the August 2009 archives and can be used
+to check whether this script works correctly:
+
+- Positive result of echelon1+2 being a relay:
+
+ $ python exonerator.py 209.17.171.104 2009-08-15 16:05:00
+ $ java -cp .:bcprov-jdk16-143.jar ExoneraTor data/ 209.17.171.104 \
+ 2009-08-15 16:05:00
+
+- Positive result of echelon1+2 exiting to google.com on any port
+
+ $ python exonerator.py 209.17.171.104 2009-08-15 16:05:00 209.85.129.104
+ $ java -cp .:bcprov-jdk16-143.jar ExoneraTor data/ 209.17.171.104 \
+ 2009-08-15 16:05:00 209.85.129.104
+
+- Positive result of echelon1+2 exiting to google.com on port 80
+
+ $ python exonerator.py 209.17.171.104 2009-08-15 16:05:00 \
+ 209.85.129.104:80
+ $ java -cp .:bcprov-jdk16-143.jar ExoneraTor data/ 209.17.171.104 \
+ 2009-08-15 16:05:00 209.85.129.104:80
+
+- Negative result of echelon1+2 exiting to google.com, but not on port 25
+
+ $ python exonerator.py 209.17.171.104 2009-08-15 16:05:00 \
+ 209.85.129.104:25
+ $ java -cp .:bcprov-jdk16-143.jar ExoneraTor data/ 209.17.171.104 \
+ 2009-08-15 16:05:00 209.85.129.104:25
+
+- Negative result with IP address of echelon1+2 changed in the last octet
+
+ $ python exonerator.py 209.17.171.50 2009-08-15 16:05:00
+ $ java -cp .:bcprov-jdk16-143.jar ExoneraTor data/ 209.17.171.50 \
+ 2009-08-15 16:05:00
+
diff --git a/exonerator/LICENSE b/exonerator/LICENSE
new file mode 100644
index 0000000..4bdb99d
--- /dev/null
+++ b/exonerator/LICENSE
@@ -0,0 +1,30 @@
+Copyright 2009 The Tor Project
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are
+met:
+
+* Redistributions of source code must retain the above copyright
+ notice, this list of conditions and the following disclaimer.
+
+* Redistributions in binary form must reproduce the above
+ copyright notice, this list of conditions and the following disclaimer
+ in the documentation and/or other materials provided with the
+ distribution.
+
+ * Neither the names of the copyright owners nor the names of its
+ contributors may be used to endorse or promote products derived from
+ this software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
+OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
+SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
+LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
+DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
+THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+
diff --git a/exonerator/exonerator.py b/exonerator/exonerator.py
new file mode 100755
index 0000000..641d65e
--- /dev/null
+++ b/exonerator/exonerator.py
@@ -0,0 +1,371 @@
+#!/usr/bin/env python
+# Copyright 2009 The Tor Project -- see LICENSE for licensing information
+
+import binascii
+import os
+import sys
+import time
+from optparse import OptionParser
+from IPy import IP
+
+USAGE = "usage: %prog [options] <IP address in question> " \
+ "<timestamp, in UTC, formatted as YYYY-MM-DD hh:mm:ss> " \
+ "[<target address>[:<target port>]]"
+DELIMITER = "-" * 75
+
+if __name__ == '__main__':
+ # check parameters
+ parser = OptionParser(usage=USAGE)
+ parser.add_option("-a", "--archive", dest="archive", default="data/",
+ help="descriptor archive directory")
+ (options, args) = parser.parse_args()
+ if len(args) not in (3, 4):
+ parser.error("incorrect number of arguments")
+ if not os.path.isdir(options.archive):
+ parser.error("descriptor archive directory %s does not exist or " \
+ "is not a directory." % \
+ os.path.abspath(options.archive))
+ archiveDirectory = os.path.dirname(options.archive)
+ try:
+ relayIP = IP(args[0])
+ except ValueError:
+ parser.error("invalid IP address in question: '%s'" % args[0])
+ timestampStr = "%s %s" % (args[1], args[2])
+ os.environ['TZ'] = 'UTC'
+ time.tzset()
+ try:
+ timestamp = time.strptime(timestampStr, "%Y-%m-%d %H:%M:%S")
+ except ValueError:
+ parser.error("incorrect time format: '%s'" % timestampStr)
+ # if a target is given, parse address and possibly port part of it
+ target = None
+ targetIP = None
+ targetPort = None
+ if len(args) == 4:
+ target = args[3]
+ targetParts = target.split(":")
+ try:
+ targetIP = IP(targetParts[0])
+ except ValueError:
+ parser.error("invalid target IP address in: '%s'" % args[3])
+ if len(targetParts) > 2:
+ parser.error("invalid target format: '%s'" % args[3])
+ if len(targetParts) > 1:
+ try:
+ targetPortTest = int(targetParts[1])
+ except ValueError:
+ parser.error("invalid target port number in: '%s'" % \
+ args[3])
+ if targetPortTest not in range(1, 65535):
+ parser.error("invalid target port number in: '%s'" % \
+ args[3])
+ targetPort = targetParts[1]
+
+ targetHelpStr = ""
+ if target:
+ targetHelpStr = " permitting exiting to %s" % target
+ print "\nTrying to find out whether %s was running a Tor relay at " \
+ "%s%s...\n\n%s\n" % (relayIP, timestampStr, targetHelpStr,
+ DELIMITER)
+
+ # check that we have the required archives
+ timestampTooOld = time.gmtime(time.mktime(timestamp) - 300 * 60)
+ timestampFrom = time.gmtime(time.mktime(timestamp) - 180 * 60)
+ timestampTooNew = time.gmtime(time.mktime(timestamp) + 120 * 60)
+ timestampTooOldStr = time.strftime("%Y-%m-%d %H:%M:%S",
+ timestampTooOld)
+ timestampFromStr = time.strftime("%Y-%m-%d %H:%M:%S", timestampFrom)
+ timestampTooNewStr = time.strftime("%Y-%m-%d %H:%M:%S",
+ timestampTooNew)
+ print "\nChecking that relevant archives between %s and %s are " \
+ "available..." % (timestampTooOldStr, timestampTooNewStr)
+
+ requiredDirs = set()
+ requiredDirs.add(time.strftime("consensuses-%Y-%m", timestampTooOld))
+ requiredDirs.add(time.strftime("consensuses-%Y-%m", timestampTooNew))
+ if target:
+ requiredDirs.add(time.strftime("server-descriptors-%Y-%m",
+ timestampTooOld))
+ requiredDirs.add(time.strftime("server-descriptors-%Y-%m",
+ timestampTooNew))
+
+ consensusDirs = list()
+ descriptorsDirs = list()
+ directoriesLeftToParse = list()
+ directoriesLeftToParse.append(archiveDirectory)
+
+ while directoriesLeftToParse:
+ directoryOrFile = directoriesLeftToParse.pop()
+ basename = os.path.basename(directoryOrFile)
+ if basename.startswith("consensuses-"):
+ if basename in requiredDirs:
+ requiredDirs.remove(basename)
+ consensusDirs.append(directoryOrFile)
+ elif basename.startswith("server-descriptors-"):
+ if basename in requiredDirs:
+ requiredDirs.remove(basename)
+ descriptorsDirs.append(directoryOrFile)
+ else:
+ for filename in os.listdir(directoryOrFile):
+ entry = "%s/%s" % (directoryOrFile, filename)
+ if os.path.isdir(entry):
+ directoriesLeftToParse.append(entry)
+
+ consensusDirs.sort()
+ for consensusDir in consensusDirs:
+ print " %s" % consensusDir
+ descriptorsDirs.sort()
+ for descriptorsDir in descriptorsDirs:
+ print " %s" % descriptorsDir
+
+ if requiredDirs:
+ print "\nWe are missing consensuses and/or server descriptors. " \
+ "Please download these archives and extract them to your " \
+ "data directory. Be sure NOT to rename the extracted " \
+ "directories or the contained files."
+ for requiredDir in sorted(requiredDirs):
+ print " %s.tar.bz2" % requiredDir
+ sys.exit()
+
+ # look for consensus files
+ print "\nLooking for relevant consensuses between %s and %s..." % \
+ (timestampFromStr, timestampStr)
+ tooOldConsensuses = set()
+ relevantConsensuses = set()
+ tooNewConsensuses = set()
+ directoriesLeftToParse = list(consensusDirs)
+ while directoriesLeftToParse:
+ directoryOrFile = directoriesLeftToParse.pop()
+ if os.path.isdir(directoryOrFile):
+ for filename in os.listdir(directoryOrFile):
+ entry = "%s/%s" % (directoryOrFile, filename)
+ directoriesLeftToParse.append(entry)
+ else:
+ basename = os.path.basename(directoryOrFile)
+ if (basename.endswith("consensus")):
+ consensusTime = time.strptime(basename[0:19],
+ "%Y-%m-%d-%H-%M-%S")
+ if consensusTime >= timestampTooOld and \
+ consensusTime < timestampFrom:
+ tooOldConsensuses.add(directoryOrFile)
+ elif consensusTime >= timestampFrom and \
+ consensusTime <= timestamp:
+ relevantConsensuses.add(directoryOrFile)
+ elif consensusTime > timestamp and \
+ consensusTime <= timestampTooNew:
+ tooNewConsensuses.add(directoryOrFile)
+ allConsensuses = set()
+ allConsensuses.update(tooOldConsensuses)
+ allConsensuses.update(relevantConsensuses)
+ allConsensuses.update(tooNewConsensuses)
+ if not allConsensuses:
+ print " None found!\n\n%s\n\nResult is INDECISIVE!\n\nWe " \
+ "cannot make any statement about IP address %s being a " \
+ "relay at %s or not! We did not find any relevant " \
+ "consensuses preceding the given time. This either means " \
+ "that you did not download and extract the consensus " \
+ "archives preceding the hours before the given time, or " \
+ "(in rare cases) that the directory archives are missing " \
+ "the hours before the timestamp. Please check that your " \
+ "directory archives contain consensus files of the " \
+ "interval 5:00 hours before and 2:00 hours after the time " \
+ "you are looking for.\n" % (DELIMITER, relayIP, timestampStr)
+ sys.exit()
+ for consensus in sorted(relevantConsensuses):
+ print " %s" % consensus
+
+ # parse consensuses to find descriptors belonging to the IP address
+ print "\nLooking for descriptor identifiers referenced in \"r \" " \
+ "lines in these consensuses containing IP address %s..." % \
+ relayIP
+ positiveConsensusesNoTarget = set()
+ addressesInSameNetwork = set()
+ relevantDescriptors = dict()
+ for consensus in allConsensuses:
+ if consensus in relevantConsensuses:
+ print " %s" % consensus
+ consensusFile = open(consensus, "r")
+ line = consensusFile.readline()
+ while line:
+ if line.startswith("r "):
+ address = IP(line.split(" ")[6])
+ if address == relayIP:
+ hexDesc = binascii.b2a_hex(binascii.a2b_base64(
+ line.split(" ")[3] + "=="))
+ if hexDesc not in relevantDescriptors.keys():
+ relevantDescriptors[hexDesc] = set()
+ relevantDescriptors[hexDesc].add(consensus)
+ positiveConsensusesNoTarget.add(consensus)
+ if consensus in relevantConsensuses:
+ print " \"%s\" references descriptor %s" % \
+ (line.rstrip(), hexDesc)
+ elif relayIP.overlaps(IP("%s/24" % address,
+ make_net=True)):
+ addressesInSameNetwork.add(address)
+ line = consensusFile.readline()
+ consensusFile.close()
+ if not relevantDescriptors:
+ print " None found!\n\n%s\n\nResult is NEGATIVE with moderate " \
+ "certainty!\n\nWe did not find IP address %s in any of " \
+ "the consensuses that were published between %s and " \
+ "%s.\n\nA possible reason for false negatives is that the " \
+ "relay is using a different IP address when generating a " \
+ "descriptor than for exiting to the Internet. We hope to " \
+ "provide better checks for this case in the future." % \
+ (DELIMITER, relayIP, timestampTooOldStr, timestampTooNewStr)
+ if addressesInSameNetwork:
+ print "\nThe following other IP addresses of Tor relays " \
+ "were found in the mentioned consensus files that are " \
+ "in the same /24 network and that could be related to " \
+ "IP address %s:" % relayIP
+ for addr in addressesInSameNetwork:
+ print " %s" % addr
+ print ""
+ sys.exit()
+
+ # parse router descriptors to check exit policies
+ positiveConsensuses = set()
+ missingDescriptors = set()
+ if target:
+ print "\nChecking if referenced descriptors permit exiting to " \
+ "%s..." % target
+ descriptors = relevantDescriptors.keys()
+ for desc in descriptors:
+ missingDescriptors.add(desc)
+ directoriesLeftToParse = list(descriptorsDirs)
+ while directoriesLeftToParse:
+ directoryOrFile = directoriesLeftToParse.pop()
+ if os.path.isdir(directoryOrFile):
+ for filename in os.listdir(directoryOrFile):
+ entry = "%s/%s" % (directoryOrFile, filename)
+ directoriesLeftToParse.append(entry)
+ else:
+ basename = os.path.basename(directoryOrFile)
+ for descriptor in descriptors:
+ if basename == descriptor:
+ missingDescriptors.remove(descriptor)
+ descriptorFile = open(directoryOrFile, "r")
+ line = descriptorFile.readline()
+ while line:
+ if line.startswith("reject ") or \
+ line.startswith("accept "):
+ ruleAccept = line.split()[0] == "accept"
+ ruleAddress = line.split()[1].split(":")[0]
+ if ruleAddress != "*" and not \
+ IP(ruleAddress).overlaps(targetIP):
+ # IP address does not match
+ line = descriptorFile.readline()
+ continue
+ rulePort = line.split()[1].split(":")[1]
+ if not targetPort and not ruleAccept and \
+ rulePort != "*":
+ # with no port given, we only consider
+ # reject :* rules as matching
+ line = descriptorFile.readline()
+ continue
+ if targetPort and rulePort != "*" and \
+ targetPort != rulePort:
+ # ports do not match
+ line = descriptorFile.readline()
+ continue
+ relevantMatch = False
+ for f in relevantDescriptors.get(
+ descriptor):
+ if f in relevantConsensuses:
+ relevantMatch = True
+ if relevantMatch:
+ if ruleAccept:
+ print " %s permits exiting to " \
+ "%s according to rule " \
+ "\"%s\"" % (directoryOrFile,
+ target, line.rstrip())
+ else:
+ print " %s does not permit " \
+ "exiting to %s according " \
+ "to rule \"%s\"" % \
+ (directoryOrFile,
+ target, line.rstrip())
+ if ruleAccept:
+ for consensus in \
+ relevantDescriptors.get(
+ descriptor):
+ positiveConsensuses.add(consensus)
+ break
+ line = descriptorFile.readline()
+ descriptorFile.close()
+
+ # print out result
+ matches = None
+ if target:
+ matches = positiveConsensuses
+ else:
+ matches = positiveConsensusesNoTarget
+ lastConsensus = sorted(relevantConsensuses)[len(relevantConsensuses)-1]
+ if lastConsensus in matches:
+ print "\n%s\n\nResult is POSITIVE with high certainty!\n\nWe " \
+ "found one or more relays on IP address %s%s in the most " \
+ "recent consensus preceding %s that clients were likely " \
+ "to know.\n" % (DELIMITER, relayIP, targetHelpStr,
+ timestampStr)
+ sys.exit()
+ resultIndecisive = target and len(missingDescriptors) > 0
+ if resultIndecisive:
+ print "\n%s\n\nResult is INDECISIVE!\n\nAt least one " \
+ "referenced descriptor could not be found. This is a rare " \
+ "case, but one that (apparently) happens. We cannot make " \
+ "any good statement about exit relays without these " \
+ "descriptors. The following descriptors are missing:" % \
+ DELIMITER
+ for desc in missingDescriptors:
+ print " %s" % desc
+ inOtherRelevantConsensus = False
+ inTooOldConsensuses = False
+ inTooNewConsensuses = False
+ for f in matches:
+ if f in relevantConsensuses:
+ inOtherRelevantConsensus = True
+ elif f in tooOldConsensuses:
+ inTooOldConsensuses = True
+ elif f in tooNewConsensuses:
+ inTooNewConsensuses = True
+ if inOtherRelevantConsensus:
+ if not resultIndecisive:
+ print "\n%s\n\nResult is POSITIVE with moderate certainty!" % \
+ DELIMITER
+ print "\nWe found one or more relays on IP address %s%s, but " \
+ "not in the consensus immediately preceding %s. A " \
+ "possible reason for the relay being missing in the last " \
+ "consensus preceding the given time might be that some of " \
+ "the directory authorities had difficulties connecting to " \
+ "the relay. However, clients might still have used the " \
+ "relay." % (relayIP, targetHelpStr, timestampStr)
+ else:
+ if not resultIndecisive:
+ print "\n%s\n\nResult is NEGATIVE with high certainty!" % \
+ DELIMITER
+ print "\nWe did not find any relay on IP address %s%s in the " \
+ "consensuses 3:00 hours preceding %s." % (relayIP,
+ targetHelpStr, timestampStr)
+ if inTooOldConsensuses or inTooNewConsensuses:
+ if inTooOldConsensuses and not inTooNewConsensuses:
+ print "\nNote that we found a matching relay in " \
+ "consensuses that were published between 5:00 and " \
+ "3:00 hours before %s." % timestampStr
+ elif not inTooOldConsensuses and inTooNewConsensuses:
+ print "\nNote that we found a matching relay in " \
+ "consensuses that were published up to 2:00 hours " \
+ "after %s." % timestampStr
+ else:
+ print "\nNote that we found a matching relay in " \
+ "consensuses that were published between 5:00 and " \
+ "3:00 hours before and in consensuses that were " \
+ "published up to 2:00 hours after %s." % timestampStr
+ print "Make sure that the timestamp you provided is in the " \
+ "correct timezone: UTC (or GMT)."
+ if target:
+ if not positiveConsensuses and positiveConsensusesNoTarget:
+ print "\nNote that although the found relay(s) did not " \
+ "permit exiting to %s there have been one or more " \
+ "relays running at the given time." % target
+ print ""
+
--
1.7.1