diff options
-rwxr-xr-x | contrib/tor-control.py | 206 |
1 files changed, 178 insertions, 28 deletions
diff --git a/contrib/tor-control.py b/contrib/tor-control.py index 929ee43b01..8c7ecb9f7a 100755 --- a/contrib/tor-control.py +++ b/contrib/tor-control.py @@ -5,13 +5,59 @@ import socket import struct import sys +MSG_TYPE_ERROR = 0x0000 +MSG_TYPE_DONE = 0x0001 MSG_TYPE_SETCONF = 0x0002 MSG_TYPE_GETCONF = 0x0003 +MSG_TYPE_CONFVALUE = 0x0004 MSG_TYPE_SETEVENTS = 0x0005 +MSG_TYPE_EVENT = 0x0006 MSG_TYPE_AUTH = 0x0007 - -EVENT_TYPE_BANDWIDTH = 0x0004 -EVENT_TYPE_WARN = 0x0005 +MSG_TYPE_SAVECONF = 0x0008 +MSG_TYPE_SIGNAL = 0x0009 +MSG_TYPE_MAPADDRESS = 0x000A +MSG_TYPE_GETINFO = 0x000B +MSG_TYPE_INFOVALUE = 0x000C +MSG_TYPE_EXTENDCIRCUIT = 0x000D +MSG_TYPE_ATTACHSTREAM = 0x000E +MSG_TYPE_POSTDESCRIPTOR = 0x000F +MSG_TYPE_FRAGMENTHEADER = 0x0010 +MSG_TYPE_FRAGMENT = 0x0011 +MSG_TYPE_REDIRECTSTREAM = 0x0012 +MSG_TYPE_CLOSESTREAM = 0x0013 +MSG_TYPE_CLOSECIRCUIT = 0x0014 + +EVENT_TYPE_CIRCSTATUS = 0x0001 +EVENT_TYPE_STREAMSTATUS = 0x0002 +EVENT_TYPE_ORCONNSTATUS = 0x0003 +EVENT_TYPE_BANDWIDTH = 0x0004 +EVENT_TYPE_WARN = 0x0005 +EVENT_TYPE_NEWDESC = 0x0006 + +ERR_CODES = { + 0x0000 : "Unspecified error", + 0x0001 : "Internal error", + 0x0002 : "Unrecognized message type", + 0x0003 : "Syntax error", + 0x0004 : "Unrecognized configuration key", + 0x0005 : "Invalid configuration value", + 0x0006 : "Unrecognized byte code", + 0x0007 : "Unauthorized", + 0x0008 : "Failed authentication attempt", + 0x0009 : "Resource exhausted", + 0x000A : "No such stream", + 0x000B : "No such circuit", + 0x000C : "No such OR" +} + +class TorCtlError(Exception): + pass + +class ProtocolError(TorCtlError): + pass + +class ErrorReply(TorCtlError): + pass def parseHostAndPort(h): host, port = "localhost", 9051 @@ -31,45 +77,134 @@ def parseHostAndPort(h): return host, port -def receive_message(s): +def _receive_msg(s): body = "" header = s.recv(4) length,type = struct.unpack("!HH",header) - print "Got response length %d, type %d"%(length,type) if length: body = s.recv(length) - print "Got response length %d, type %d, body %s"%(length,type,body) return length,type,body +def receive_message(s): + length, tp, body = _receive_msg(s) + if tp != MSG_TYPE_FRAGMENTHEADER: + return length, tp, body + if length < 6: + raise ProtocolError("FRAGMENTHEADER message too short") + realType,realLength = struct.unpack("!HL", body[:6]) + data = [ body[6:] ] + soFar = len(data[0]) + while 1: + length, tp, body = _receive_msg(s) + if tp != MSG_TYPE_FRAGMENT: + raise ProtocolError("Missing FRAGMENT message") + soFar += length + data.append(body) + if soFar == realLength: + return realLength, realType, "".join(data) + elif soFar > realLengtH: + raise ProtocolError("FRAGMENT message too long!") + +_event_handler = None +def receive_reply(s, expected=None): + while 1: + _, tp, body = receive_message(s) + if tp == MSG_TYPE_EVENT: + if _event_handler is not None: + _event_handler(tp, body) + elif tp == MSG_TYPE_ERROR: + if len(body)<2: + raise ProtocolError("(Truncated error message)") + errCode, = struct.unpack("!H", body[:2]) + raise ErrorReply((errCode, + ERR_CODES.get(errCode,"[unrecognized]"), + body[2:])) + elif (expected is not None) and (tp not in expected): + raise ProtocolError("Unexpected message type 0x%04x"%tp) + else: + return tp, body + def pack_message(type, body=""): length = len(body) - reqheader = struct.pack("!HH", length, type) - return "%s%s"%(reqheader,body) + if length < 65536: + reqheader = struct.pack("!HH", length, type) + return "%s%s"%(reqheader,body) + + fragheader = struct.pack("!HHHL", + 65535, MSG_TYPE_FRAGMENTHEADER, type, length) + msgs = [ fragheader, body[:65535-6] ] + body = body[65535-6:] + while body: + if len(body) > 65535: + fl = 65535 + else: + fl = len(body) + fragheader = struct.pack("!HH", MSG_TYPE_FRAGMENT, fl) + msgs.append(fragheader) + msgs.append(body[:fl]) + body = body[fl:] + + return "".join(msgs) + +def send_message(s, type, body=""): + s.sendall(pack_message(type, body)) def authenticate(s): - s.sendall(pack_message(MSG_TYPE_AUTH)) - length,type,body = receive_message(s) + send_message(s,MSG_TYPE_AUTH) + type,body = receive_reply(s) return +def _parseKV(body,sep=" ",term="\n"): + res = [] + for line in body.split(term): + if not line: continue + print repr(line) + k, v = line.split(sep,1) + res.append((k,v)) + return res + def get_option(s,name): - s.sendall(pack_message(MSG_TYPE_GETCONF,name)) - length,type,body = receive_message(s) - return + send_message(s,MSG_TYPE_GETCONF,name) + tp,body = receive_reply(s,[MSG_TYPE_CONFVALUE]) + return _parseKV(body) def set_option(s,msg): - s.sendall(pack_message(MSG_TYPE_SETCONF,msg)) - length,type,body = receive_message(s) + send_message(s,MSG_TYPE_SETCONF,msg) + tp,body = receive_reply(s,[MSG_TYPE_DONE]) + +def get_info(s,name): + send_message(s,MSG_TYPE_GETINFO,name) + tp,body = receive_reply(s,[MSG_TYPE_INFOVALUE]) + kvs = body.split("\0") + d = {} + for i in xrange(0,len(kvs)-1,2): + d[kvs[i]] = kvs[i+1] + return d + +def set_events(s,events): + send_message(s,MSG_TYPE_SETEVENTS, + "".join([struct.pack("!H", event) for event in events])) + type,body = receive_reply(s,[MSG_TYPE_DONE]) return -def get_event(s,events): - eventbody = struct.pack("!H", events) - s.sendall(pack_message(MSG_TYPE_SETEVENTS,eventbody)) - length,type,body = receive_message(s) - return +def save_conf(s): + send_message(s,MSG_TYPE_SAVECONF) + receive_reply(s,[MSG_TYPE_DONE]) + +def send_signal(s, sig): + send_message(s,MSG_TYPE_SIGNAL,struct.pack("B",sig)) + receive_reply(s,[MSG_TYPE_DONE]) + +def map_address(s, kv): + msg = [ "%s %s\n"%(k,v) for k,v in kv ] + send_message(s,MSG_TYPE_MAPADDRESS,"".join(msg)) + tp, body = receive_reply(s,[MSG_TYPE_DONE]) + return _parseKV(body) def listen_for_events(s): while(1): - length,type,body = receive_message(s) + _,type,body = receive_message(s) + print "event",type return def do_main_loop(host,port): @@ -77,13 +212,28 @@ def do_main_loop(host,port): s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) s.connect((host,port)) authenticate(s) - get_option(s,"nickname") - get_option(s,"DirFetchPostPeriod\n") - set_option(s,"1") - set_option(s,"bandwidthburstbytes 100000") -# set_option(s,"runasdaemon 1") -# get_event(s,EVENT_TYPE_WARN) - get_event(s,EVENT_TYPE_BANDWIDTH) + print "nick",`get_option(s,"nickname")` + print get_option(s,"DirFetchPeriod\n") + print `get_info(s,"version")` + #print `get_info(s,"desc/name/moria1")` + #print `get_info(s,"network-status")` + print `get_info(s,"addr-mappings/all")` + print `get_info(s,"addr-mappings/config")` + print `get_info(s,"addr-mappings/cache")` + print `get_info(s,"addr-mappings/control")` + print `map_address(s, [("0.0.0.0", "Foobar.com"), + ("1.2.3.4", "foobaz.com"), + ("frebnitz.com", "5.6.7.8"), + (".", "abacinator.onion")])` + send_signal(s,1) + #save_conf(s) + + + #set_option(s,"1") + #set_option(s,"bandwidthburstbytes 100000") + #set_option(s,"runasdaemon 1") + #set_events(s,[EVENT_TYPE_WARN]) + set_events(s,[EVENT_TYPE_WARN,EVENT_TYPE_STREAMSTATUS]) listen_for_events(s) |