2016-06-04 06:07:50 +00:00
|
|
|
from __future__ import print_function, unicode_literals
|
2017-05-23 07:45:02 +00:00
|
|
|
import sys
|
|
|
|
from zope.interface.declarations import directlyProvides
|
2016-03-03 20:24:07 +00:00
|
|
|
from twisted.internet.defer import inlineCallbacks, returnValue
|
rewrite Tor support (py2 only)
The new TorManager adds --launch-tor and --tor-control-port= arguments
(requiring the user to explicitly request a new Tor process, if that's what
they want). The default (when --tor is enabled) looks for a control port in
the usual places (/var/run/tor/control, localhost:9051, localhost:9151), then
falls back to hoping there's a SOCKS port in the usual
place (localhost:9050). (closes #64)
The ssh utilities should now accept the same tor arguments as ordinary
send/receive commands. There are now full tests for TorManager, and basic
tests for how send/receive use it. (closes #97)
Note that Tor is only supported on python2.7 for now, since txsocksx (and
therefore txtorcon) doesn't work on py3. You need to do "pip install
magic-wormhole[tor]" to get Tor support, and that will get you an inscrutable
error on py3 (referencing vcversioner, "install_requires must be a string or
list of strings", and "int object not iterable").
To run tests, you must install with the [dev] extra (to get "mock" and other
libraries). Our setup.py only includes "txtorcon" in the [dev] extra when on
py2, not on py3. Unit tests tolerate the lack of txtorcon (they mock out
everything txtorcon would provide), so they should provide the same coverage
on both py2 and py3.
2017-01-16 03:24:23 +00:00
|
|
|
try:
|
2017-05-23 07:45:02 +00:00
|
|
|
import txtorcon
|
rewrite Tor support (py2 only)
The new TorManager adds --launch-tor and --tor-control-port= arguments
(requiring the user to explicitly request a new Tor process, if that's what
they want). The default (when --tor is enabled) looks for a control port in
the usual places (/var/run/tor/control, localhost:9051, localhost:9151), then
falls back to hoping there's a SOCKS port in the usual
place (localhost:9050). (closes #64)
The ssh utilities should now accept the same tor arguments as ordinary
send/receive commands. There are now full tests for TorManager, and basic
tests for how send/receive use it. (closes #97)
Note that Tor is only supported on python2.7 for now, since txsocksx (and
therefore txtorcon) doesn't work on py3. You need to do "pip install
magic-wormhole[tor]" to get Tor support, and that will get you an inscrutable
error on py3 (referencing vcversioner, "install_requires must be a string or
list of strings", and "int object not iterable").
To run tests, you must install with the [dev] extra (to get "mock" and other
libraries). Our setup.py only includes "txtorcon" in the [dev] extra when on
py2, not on py3. Unit tests tolerate the lack of txtorcon (they mock out
everything txtorcon would provide), so they should provide the same coverage
on both py2 and py3.
2017-01-16 03:24:23 +00:00
|
|
|
except ImportError:
|
2017-05-23 07:45:02 +00:00
|
|
|
txtorcon = None
|
|
|
|
from . import _interfaces, errors
|
2016-05-24 23:25:12 +00:00
|
|
|
from .timing import DebugTiming
|
2016-03-03 20:24:07 +00:00
|
|
|
|
2017-05-23 07:45:02 +00:00
|
|
|
@inlineCallbacks
|
|
|
|
def get_tor(reactor, launch_tor=False, tor_control_port=None,
|
|
|
|
timing=None, stderr=sys.stderr):
|
|
|
|
"""
|
|
|
|
If launch_tor=True, I will try to launch a new Tor process, ask it
|
|
|
|
for its SOCKS and control ports, and use those for outbound
|
|
|
|
connections (and inbound onion-service listeners, if necessary).
|
|
|
|
|
|
|
|
Otherwise if tor_control_port is provided, I will attempt to connect
|
|
|
|
to an existing Tor's control port at the endpoint it specifies. I'll
|
|
|
|
ask that Tor for its SOCKS port.
|
|
|
|
|
|
|
|
With no arguments, I will try to connect to an existing Tor's control
|
|
|
|
port at the usual places: [unix:/var/run/tor/control,
|
|
|
|
tcp:127.0.0.1:9051, tcp:127.0.0.1:9151]. If any are successful, I'll
|
|
|
|
ask that Tor for its SOCKS port. If none are successful, I'll attempt
|
|
|
|
to do SOCKS to tcp:127.0.0.1:9050.
|
|
|
|
|
|
|
|
If I am unable to make a SOCKS connection, the initial connection to
|
|
|
|
the Rendezvous Server will fail, and the program will terminate.
|
|
|
|
|
|
|
|
Control-port connections can only succeed if I can authenticate (by
|
|
|
|
reading a cookie file named by the Tor process), so the current user
|
|
|
|
must have permission to read that file (either they started Tor, e.g.
|
|
|
|
TorBrowser, or they are in a unix group that's been given access,
|
|
|
|
e.g. debian-tor).
|
|
|
|
"""
|
|
|
|
# rationale: launching a new Tor takes a long time, so only do it if
|
|
|
|
# the user specifically asks for it with --launch-tor. Using an
|
|
|
|
# existing Tor should be much faster, but still requires general
|
|
|
|
# permission via --tor.
|
|
|
|
|
|
|
|
if not txtorcon:
|
|
|
|
raise errors.NoTorError()
|
|
|
|
|
|
|
|
if not isinstance(launch_tor, bool): # note: False is int
|
|
|
|
raise TypeError("launch_tor= must be boolean")
|
|
|
|
if not isinstance(tor_control_port, (type(""), type(None))):
|
|
|
|
raise TypeError("tor_control_port= must be str or None")
|
|
|
|
assert tor_control_port != ""
|
|
|
|
if launch_tor and tor_control_port is not None:
|
|
|
|
raise ValueError("cannot combine --launch-tor and --tor-control-port=")
|
|
|
|
timing = timing or DebugTiming()
|
|
|
|
|
|
|
|
# Connect to an existing Tor, or create a new one. If we need to
|
|
|
|
# launch an onion service, then we need a working control port (and
|
|
|
|
# authentication cookie). If we're only acting as a client, we don't
|
|
|
|
# need the control port.
|
|
|
|
|
|
|
|
if launch_tor:
|
|
|
|
print(" launching a new Tor process, this may take a while..",
|
|
|
|
file=stderr)
|
|
|
|
with timing.add("launch tor"):
|
|
|
|
tor = yield txtorcon.launch(reactor,
|
|
|
|
#data_directory=,
|
|
|
|
#tor_binary=,
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
with timing.add("find tor"):
|
|
|
|
try:
|
|
|
|
tor = yield txtorcon.connect(reactor, tor_control_port)
|
|
|
|
print(" using Tor", file=stderr)
|
|
|
|
except Exception:
|
|
|
|
#socks_desc = "tcp:127.0.0.1:9050" # fallback
|
|
|
|
#print(" using Tor (SOCKS port %s)" % socks_desc,
|
|
|
|
# file=stderr)
|
|
|
|
print(" unable to find control port, bailing",
|
|
|
|
file=stderr)
|
|
|
|
# TODO: something nicer. I think connect() is likely to throw
|
|
|
|
# a reactor.connectTCP -type error, like ConnectionFailed or
|
|
|
|
# ConnectionRefused or something
|
|
|
|
raise
|
|
|
|
directlyProvides(tor, _interfaces.ITorManager)
|
|
|
|
returnValue(tor)
|