[tor-commits] [check/master] Start an API.

arlo at torproject.org arlo at torproject.org
Tue Nov 19 08:03:12 UTC 2013


commit 940a0ab0e048913272e4bbfa1b75ac14a456b33e
Author: Arlo Breault <arlolra at gmail.com>
Date:   Tue Nov 19 00:00:59 2013 -0800

    Start an API.
    
    One route, /api/bulk
---
 check.go     |    1 +
 datastore.go |   36 ++++++++++++++++++++++++++++++++----
 handlers.go  |   18 +++++++++++++-----
 3 files changed, 46 insertions(+), 9 deletions(-)

diff --git a/check.go b/check.go
index 73e6a7c..00b25b6 100644
--- a/check.go
+++ b/check.go
@@ -60,6 +60,7 @@ func main() {
 	// routes
 	http.HandleFunc("/", RootHandler(CompileTemplate(*basePath, domain, "index.html"), exits, domain, Phttp, Locales))
 	bulk := BulkHandler(CompileTemplate(*basePath, domain, "bulk.html"), exits, domain)
+	http.HandleFunc("/api/bulk", bulk)
 	http.HandleFunc("/torbulkexitlist", bulk)
 	http.HandleFunc("/cgi-bin/TorBulkExitList.py", bulk)
 
diff --git a/datastore.go b/datastore.go
index c01acc6..c23109b 100644
--- a/datastore.go
+++ b/datastore.go
@@ -84,6 +84,19 @@ func (p PolicyList) Swap(i, j int) {
 	p[i], p[j] = p[j], p[i]
 }
 
+type ExitInfo struct {
+	Address     string
+	Fingerprint string
+}
+
+func (e ExitInfo) toJSON() (b []byte) {
+	j, err := json.MarshalIndent(e, "", "  ")
+	if err != nil {
+		return
+	}
+	return j
+}
+
 type Exits struct {
 	List        PolicyList
 	UpdateTime  time.Time
@@ -93,15 +106,30 @@ type Exits struct {
 
 func (e *Exits) Dump(w io.Writer, tminus int, ip string, port int) {
 	ap := AddressPort{ip, port}
-	e.GetAllExits(ap, tminus, func(exit string, _ string) {
+	e.GetAllExits(ap, tminus, func(exit string, _ string, _ int) {
 		w.Write([]byte(exit + "\n"))
 	})
 }
 
-func (e *Exits) GetAllExits(ap AddressPort, tminus int, fn func(string, string)) {
+func (e *Exits) DumpJSON(w io.Writer, tminus int, ip string, port int) {
+	ap := AddressPort{ip, port}
+	Prefix := []byte(",\n")
+	w.Write([]byte("["))
+	e.GetAllExits(ap, tminus, func(address string, fingerprint string, ind int) {
+		if ind > 0 {
+			w.Write(Prefix)
+		}
+		w.Write(ExitInfo{address, fingerprint}.toJSON())
+	})
+	w.Write([]byte("]"))
+}
+
+func (e *Exits) GetAllExits(ap AddressPort, tminus int, fn func(string, string, int)) {
+	ind := 0
 	for _, val := range e.List {
 		if val.Tminus <= tminus && val.CanExit(ap) {
-			fn(val.Address, val.Fingerprint)
+			fn(val.Address, val.Fingerprint, ind)
+			ind += 1
 		}
 	}
 }
@@ -110,7 +138,7 @@ var DefaultTarget = AddressPort{"38.229.70.31", 443}
 
 func (e *Exits) PreComputeTorList() {
 	newmap := make(map[string]string)
-	e.GetAllExits(DefaultTarget, 16, func(ip string, fingerprint string) {
+	e.GetAllExits(DefaultTarget, 16, func(ip string, fingerprint string, _ int) {
 		newmap[ip] = fingerprint
 	})
 	e.IsTorLookup = newmap
diff --git a/handlers.go b/handlers.go
index eee6948..9342ff8 100644
--- a/handlers.go
+++ b/handlers.go
@@ -9,6 +9,7 @@ import (
 	"log"
 	"net"
 	"net/http"
+	"regexp"
 	"strconv"
 	"time"
 )
@@ -89,6 +90,8 @@ func RootHandler(Layout *template.Template, Exits *Exits, domain *gettext.Domain
 
 func BulkHandler(Layout *template.Template, Exits *Exits, domain *gettext.Domain) http.HandlerFunc {
 
+	ApiPath := regexp.MustCompile("^/api/")
+
 	return func(w http.ResponseWriter, r *http.Request) {
 		q := r.URL.Query()
 
@@ -103,12 +106,17 @@ func BulkHandler(Layout *template.Template, Exits *Exits, domain *gettext.Domain
 
 		w.Header().Set("Last-Modified", Exits.UpdateTime.UTC().Format(http.TimeFormat))
 
-		str := fmt.Sprintf("# This is a list of all Tor exit nodes from the past %d hours that can contact %s on port %d #\n", n, ip, port)
-		str += fmt.Sprintf("# You can update this list by visiting https://check.torproject.org/cgi-bin/TorBulkExitList.py?ip=%s%s%s #\n", ip, port_str, n_str)
-		str += fmt.Sprintf("# This file was generated on %v #\n", Exits.UpdateTime.UTC().Format(time.UnixDate))
-		fmt.Fprintf(w, str)
+		if q.Get("format") == "json" || ApiPath.MatchString(r.URL.Path) {
+			w.Header().Set("Content-Type", "application/json")
+			Exits.DumpJSON(w, n, ip, port)
+		} else {
+			str := fmt.Sprintf("# This is a list of all Tor exit nodes from the past %d hours that can contact %s on port %d #\n", n, ip, port)
+			str += fmt.Sprintf("# You can update this list by visiting https://check.torproject.org/cgi-bin/TorBulkExitList.py?ip=%s%s%s #\n", ip, port_str, n_str)
+			str += fmt.Sprintf("# This file was generated on %v #\n", Exits.UpdateTime.UTC().Format(time.UnixDate))
+			fmt.Fprintf(w, str)
+			Exits.Dump(w, n, ip, port)
+		}
 
-		Exits.Dump(w, n, ip, port)
 	}
 
 }



More information about the tor-commits mailing list