Merge pull request #1478 from mig5/fetch-builtin-bridges

Fetch the built-in bridges from Tor's Censorship Circumvention API, rather than hardcode them
This commit is contained in:
Micah Lee 2021-12-04 20:28:04 -08:00 committed by GitHub
commit 27ee69a220
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
8 changed files with 189 additions and 70 deletions

View file

@ -25,21 +25,46 @@ from .meek import MeekNotRunning
class CensorshipCircumvention(object): class CensorshipCircumvention(object):
""" """
Connect to the Tor Moat APIs to retrieve censorship Connect to the Tor Moat APIs to retrieve censorship
circumvention recommendations, over the Meek client. circumvention recommendations or the latest bridges.
We support reaching this API over Tor, or Meek
(domain fronting) if Tor is not connected.
""" """
def __init__(self, common, meek, domain_fronting=True): def __init__(self, common, meek=None, onion=None):
""" """
Set up the CensorshipCircumvention object to hold Set up the CensorshipCircumvention object to hold
common and meek objects. common and meek objects.
""" """
self.common = common self.common = common
self.meek = meek
self.common.log("CensorshipCircumvention", "__init__") self.common.log("CensorshipCircumvention", "__init__")
self.api_proxies = {}
# Bail out if we requested domain fronting but we can't use meek if meek:
if domain_fronting and not self.meek.meek_proxies: self.meek = meek
raise MeekNotRunning() if not self.meek.meek_proxies:
raise MeekNotRunning()
else:
self.common.log(
"CensorshipCircumvention",
"__init__",
"Using Meek with CensorShipCircumvention API",
)
self.api_proxies = self.meek.meek_proxies
if onion:
self.onion = onion
if not self.onion.is_authenticated:
return False
else:
self.common.log(
"CensorshipCircumvention",
"__init__",
"Using Tor with CensorShipCircumvention API",
)
(socks_address, socks_port) = self.onion.get_tor_socks_port()
self.api_proxies = {
"http": f"socks5h://{socks_address}:{socks_port}",
"https": f"socks5h://{socks_address}:{socks_port}",
}
def request_map(self, country=False): def request_map(self, country=False):
""" """
@ -52,6 +77,8 @@ class CensorshipCircumvention(object):
Note that this API endpoint doesn't return actual bridges, Note that this API endpoint doesn't return actual bridges,
it just returns the recommended bridge type countries. it just returns the recommended bridge type countries.
""" """
if not self.api_proxies:
return False
endpoint = "https://bridges.torproject.org/moat/circumvention/map" endpoint = "https://bridges.torproject.org/moat/circumvention/map"
data = {} data = {}
if country: if country:
@ -61,7 +88,7 @@ class CensorshipCircumvention(object):
endpoint, endpoint,
json=data, json=data,
headers={"Content-Type": "application/vnd.api+json"}, headers={"Content-Type": "application/vnd.api+json"},
proxies=self.meek.meek_proxies, proxies=self.api_proxies,
) )
if r.status_code != 200: if r.status_code != 200:
self.common.log( self.common.log(
@ -95,6 +122,8 @@ class CensorshipCircumvention(object):
Optionally, a list of transports can be specified in order to Optionally, a list of transports can be specified in order to
return recommended settings for just that transport type. return recommended settings for just that transport type.
""" """
if not self.api_proxies:
return False
endpoint = "https://bridges.torproject.org/moat/circumvention/settings" endpoint = "https://bridges.torproject.org/moat/circumvention/settings"
data = {} data = {}
if country: if country:
@ -105,7 +134,7 @@ class CensorshipCircumvention(object):
endpoint, endpoint,
json=data, json=data,
headers={"Content-Type": "application/vnd.api+json"}, headers={"Content-Type": "application/vnd.api+json"},
proxies=self.meek.meek_proxies, proxies=self.api_proxies,
) )
if r.status_code != 200: if r.status_code != 200:
self.common.log( self.common.log(
@ -142,11 +171,13 @@ class CensorshipCircumvention(object):
""" """
Retrieves the list of built-in bridges from the Tor Project. Retrieves the list of built-in bridges from the Tor Project.
""" """
if not self.api_proxies:
return False
endpoint = "https://bridges.torproject.org/moat/circumvention/builtin" endpoint = "https://bridges.torproject.org/moat/circumvention/builtin"
r = requests.post( r = requests.post(
endpoint, endpoint,
headers={"Content-Type": "application/vnd.api+json"}, headers={"Content-Type": "application/vnd.api+json"},
proxies=self.meek.meek_proxies, proxies=self.api_proxies,
) )
if r.status_code != 200: if r.status_code != 200:
self.common.log( self.common.log(

View file

@ -18,17 +18,19 @@ You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>. along with this program. If not, see <http://www.gnu.org/licenses/>.
""" """
from .censorship import CensorshipCircumvention
from .meek import Meek
from stem.control import Controller from stem.control import Controller
from stem import ProtocolError, SocketClosed from stem import ProtocolError, SocketClosed
from stem.connection import MissingPassword, UnreadableCookieFile, AuthenticationFailure from stem.connection import MissingPassword, UnreadableCookieFile, AuthenticationFailure
import base64 import base64
import nacl.public import nacl.public
import os import os
import tempfile
import subprocess
import time
import shlex
import psutil import psutil
import shlex
import subprocess
import tempfile
import time
import traceback import traceback
from distutils.version import LooseVersion as Version from distutils.version import LooseVersion as Version
@ -258,9 +260,7 @@ class Onion(object):
and cmdline[2] == self.tor_torrc and cmdline[2] == self.tor_torrc
): ):
self.common.log( self.common.log(
"Onion", "Onion", "connect", "found a stale tor process, killing it"
"connect",
"found a stale tor process, killing it",
) )
proc.terminate() proc.terminate()
proc.wait() proc.wait()
@ -317,49 +317,75 @@ class Onion(object):
) )
with open(self.tor_torrc, "w") as f: with open(self.tor_torrc, "w") as f:
self.common.log("Onion", "connect", "Writing torrc template file")
f.write(torrc_template) f.write(torrc_template)
# Bridge support # Bridge support
if self.settings.get("bridges_enabled"): if self.settings.get("bridges_enabled"):
f.write("\nUseBridges 1\n")
if self.settings.get("bridges_type") == "built-in": if self.settings.get("bridges_type") == "built-in":
if self.settings.get("bridges_builtin_pt") == "obfs4": use_torrc_bridge_templates = False
with open( builtin_bridge_type = self.settings.get("bridges_builtin_pt")
self.common.get_resource_path("torrc_template-obfs4") # Use built-inbridges stored in settings, if they are there already.
) as o: # They are probably newer than that of our hardcoded copies.
f.write(o.read()) if self.settings.get("bridges_builtin"):
elif self.settings.get("bridges_builtin_pt") == "meek-azure": try:
with open( for line in self.settings.get("bridges_builtin")[
self.common.get_resource_path( builtin_bridge_type
"torrc_template-meek_lite_azure" ]:
if line.strip() != "":
f.write(f"Bridge {line}\n")
self.common.log(
"Onion",
"connect",
"Wrote in the built-in bridges from OnionShare settings",
) )
) as o: except KeyError:
f.write(o.read()) # Somehow we had built-in bridges in our settings, but
elif self.settings.get("bridges_builtin_pt") == "snowflake": # not for this bridge type. Fall back to using the hard-
with open( # coded templates.
self.common.get_resource_path( use_torrc_bridge_templates = True
"torrc_template-snowflake" else:
) use_torrc_bridge_templates = True
) as o: if use_torrc_bridge_templates:
f.write(o.read()) if builtin_bridge_type == "obfs4":
with open(
self.common.get_resource_path(
"torrc_template-obfs4"
)
) as o:
f.write(o.read())
elif builtin_bridge_type == "meek-azure":
with open(
self.common.get_resource_path(
"torrc_template-meek_lite_azure"
)
) as o:
f.write(o.read())
elif builtin_bridge_type == "snowflake":
with open(
self.common.get_resource_path(
"torrc_template-snowflake"
)
) as o:
f.write(o.read())
self.common.log(
"Onion",
"connect",
"Wrote in the built-in bridges from torrc templates",
)
elif self.settings.get("bridges_type") == "moat": elif self.settings.get("bridges_type") == "moat":
for line in self.settings.get("bridges_moat").split("\n"): for line in self.settings.get("bridges_moat").split("\n"):
if line.strip() != "": if line.strip() != "":
f.write(f"Bridge {line}\n") f.write(f"Bridge {line}\n")
f.write("\nUseBridges 1\n")
elif self.settings.get("bridges_type") == "custom": elif self.settings.get("bridges_type") == "custom":
for line in self.settings.get("bridges_custom").split("\n"): for line in self.settings.get("bridges_custom").split("\n"):
if line.strip() != "": if line.strip() != "":
f.write(f"Bridge {line}\n") f.write(f"Bridge {line}\n")
f.write("\nUseBridges 1\n")
# Execute a tor subprocess # Execute a tor subprocess
self.common.log( self.common.log("Onion", "connect", f"starting {self.tor_path} subprocess")
"Onion",
"connect",
f"starting {self.tor_path} subprocess",
)
start_ts = time.time() start_ts = time.time()
if self.common.platform == "Windows": if self.common.platform == "Windows":
# In Windows, hide console window when opening tor.exe subprocess # In Windows, hide console window when opening tor.exe subprocess
@ -385,19 +411,11 @@ class Onion(object):
) )
# Wait for the tor controller to start # Wait for the tor controller to start
self.common.log( self.common.log("Onion", "connect", f"tor pid: {self.tor_proc.pid}")
"Onion",
"connect",
f"tor pid: {self.tor_proc.pid}",
)
time.sleep(2) time.sleep(2)
# Connect to the controller # Connect to the controller
self.common.log( self.common.log("Onion", "connect", "authenticating to tor controller")
"Onion",
"connect",
"authenticating to tor controller",
)
try: try:
if ( if (
self.common.platform == "Windows" self.common.platform == "Windows"
@ -638,6 +656,14 @@ class Onion(object):
# https://trac.torproject.org/projects/tor/ticket/28619 # https://trac.torproject.org/projects/tor/ticket/28619
self.supports_v3_onions = self.tor_version >= Version("0.3.5.7") self.supports_v3_onions = self.tor_version >= Version("0.3.5.7")
# Now that we are connected to Tor, if we are using built-in bridges,
# update them with the latest copy available from the Tor API
if (
self.settings.get("bridges_enabled")
and self.settings.get("bridges_type") == "built-in"
):
self.update_builtin_bridges()
def is_authenticated(self): def is_authenticated(self):
""" """
Returns True if the Tor connection is still working, or False otherwise. Returns True if the Tor connection is still working, or False otherwise.
@ -881,3 +907,68 @@ class Onion(object):
return ("127.0.0.1", 9150) return ("127.0.0.1", 9150)
else: else:
return (self.settings.get("socks_address"), self.settings.get("socks_port")) return (self.settings.get("socks_address"), self.settings.get("socks_port"))
def update_builtin_bridges(self):
"""
Use the CensorshipCircumvention API to fetch the latest built-in bridges
and update them in settings.
"""
builtin_bridges = False
meek = None
# Try obtaining bridges over Tor, if we're connected to it.
if self.is_authenticated:
self.common.log(
"Onion",
"update_builtin_bridges",
"Updating the built-in bridges. Trying over Tor first",
)
self.censorship_circumvention = CensorshipCircumvention(
self.common, None, self
)
builtin_bridges = self.censorship_circumvention.request_builtin_bridges()
if not builtin_bridges:
# Tor was not running or it failed to hit the Tor API.
# Fall back to using Meek (domain-fronting).
self.common.log(
"Onion",
"update_builtin_bridges",
"Updating the built-in bridges. Trying via Meek (no Tor)",
)
meek = Meek(self.common)
meek.start()
self.censorship_circumvention = CensorshipCircumvention(
self.common, meek, None
)
builtin_bridges = self.censorship_circumvention.request_builtin_bridges()
meek.cleanup()
if builtin_bridges:
# If we got to this point, we have bridges
self.common.log(
"Onion",
"update_builtin_bridges",
f"Obtained bridges: {builtin_bridges}",
)
if builtin_bridges["meek"]:
# Meek bridge needs to be defined as "meek_lite", not "meek",
# for it to work with obfs4proxy.
# We also refer to this bridge type as 'meek-azure' in our settings.
# So first, rename the key in the dict
builtin_bridges["meek-azure"] = builtin_bridges.pop("meek")
new_meek_bridges = []
# Now replace the values. They also need the url/front params appended
for item in builtin_bridges["meek-azure"]:
newline = item.replace("meek", "meek_lite")
new_meek_bridges.append(
f"{newline} url=https://meek.azureedge.net/ front=ajax.aspnetcdn.com"
)
builtin_bridges["meek-azure"] = new_meek_bridges
# Save the new settings
self.settings.set("bridges_builtin", builtin_bridges)
self.settings.save()
else:
self.common.log(
"Onion", "update_builtin_bridges", "Error getting built-in bridges"
)
return False

View file

@ -1,3 +1,2 @@
# Enable built-in meek-azure bridge # Enable built-in meek-azure bridge
Bridge meek_lite 0.0.2.0:3 97700DFE9F483596DDA6264C4D7DF7641E1E39CE url=https://meek.azureedge.net/ front=ajax.aspnetcdn.com Bridge meek_lite 0.0.2.0:3 97700DFE9F483596DDA6264C4D7DF7641E1E39CE url=https://meek.azureedge.net/ front=ajax.aspnetcdn.com
UseBridges 1

View file

@ -1,17 +1,16 @@
# Enable built-in obfs4-bridge # Enable built-in obfs4-bridge
Bridge obfs4 192.95.36.142:443 CDF2E852BF539B82BD10E27E9115A31734E378C2 cert=qUVQ0srL1JI/vO6V6m/24anYXiJD3QP2HgzUKQtQ7GRqqUvs7P+tG43RtAqdhLOALP7DJQ iat-mode=1
Bridge obfs4 38.229.1.78:80 C8CBDB2464FC9804A69531437BCF2BE31FDD2EE4 cert=Hmyfd2ev46gGY7NoVxA9ngrPF2zCZtzskRTzoWXbxNkzeVnGFPWmrTtILRyqCTjHR+s9dg iat-mode=1
Bridge obfs4 38.229.33.83:80 0BAC39417268B96B9F514E7F63FA6FBA1A788955 cert=VwEFpk9F/UN9JED7XpG1XOjm/O8ZCXK80oPecgWnNDZDv5pdkhq1OpbAH0wNqOT6H6BmRQ iat-mode=1 Bridge obfs4 38.229.33.83:80 0BAC39417268B96B9F514E7F63FA6FBA1A788955 cert=VwEFpk9F/UN9JED7XpG1XOjm/O8ZCXK80oPecgWnNDZDv5pdkhq1OpbAH0wNqOT6H6BmRQ iat-mode=1
Bridge obfs4 37.218.245.14:38224 D9A82D2F9C2F65A18407B1D2B764F130847F8B5D cert=bjRaMrr1BRiAW8IE9U5z27fQaYgOhX1UCmOpg2pFpoMvo6ZgQMzLsaTzzQNTlm7hNcb+Sg iat-mode=0
Bridge obfs4 85.31.186.98:443 011F2599C0E9B27EE74B353155E244813763C3E5 cert=ayq0XzCwhpdysn5o0EyDUbmSOx3X/oTEbzDMvczHOdBJKlvIdHHLJGkZARtT4dcBFArPPg iat-mode=0
Bridge obfs4 85.31.186.26:443 91A6354697E6B02A386312F68D82CF86824D3606 cert=PBwr+S8JTVZo6MPdHnkTwXJPILWADLqfMGoVvhZClMq/Urndyd42BwX9YFJHZnBB3H0XCw iat-mode=0
Bridge obfs4 144.217.20.138:80 FB70B257C162BF1038CA669D568D76F5B7F0BABB cert=vYIV5MgrghGQvZPIi1tJwnzorMgqgmlKaB77Y3Z9Q/v94wZBOAXkW+fdx4aSxLVnKO+xNw iat-mode=0
Bridge obfs4 193.11.166.194:27015 2D82C2E354D531A68469ADF7F878FA6060C6BACA cert=4TLQPJrTSaDffMK7Nbao6LC7G9OW/NHkUwIdjLSS3KYf0Nv4/nQiiI8dY2TcsQx01NniOg iat-mode=0 Bridge obfs4 193.11.166.194:27015 2D82C2E354D531A68469ADF7F878FA6060C6BACA cert=4TLQPJrTSaDffMK7Nbao6LC7G9OW/NHkUwIdjLSS3KYf0Nv4/nQiiI8dY2TcsQx01NniOg iat-mode=0
Bridge obfs4 193.11.166.194:27020 86AC7B8D430DAC4117E9F42C9EAED18133863AAF cert=0LDeJH4JzMDtkJJrFphJCiPqKx7loozKN7VNfuukMGfHO0Z8OGdzHVkhVAOfo1mUdv9cMg iat-mode=0 Bridge obfs4 85.31.186.98:443 011F2599C0E9B27EE74B353155E244813763C3E5 cert=ayq0XzCwhpdysn5o0EyDUbmSOx3X/oTEbzDMvczHOdBJKlvIdHHLJGkZARtT4dcBFArPPg iat-mode=0
Bridge obfs4 193.11.166.194:27025 1AE2C08904527FEA90C4C4F8C1083EA59FBC6FAF cert=ItvYZzW5tn6v3G4UnQa6Qz04Npro6e81AP70YujmK/KXwDFPTs3aHXcHp4n8Vt6w/bv8cA iat-mode=0
Bridge obfs4 209.148.46.65:443 74FAD13168806246602538555B5521A0383A1875 cert=ssH+9rP8dG2NLDN2XuFw63hIO/9MNNinLmxQDpVa+7kTOa9/m+tGWT1SmSYpQ9uTBGa6Hw iat-mode=0
Bridge obfs4 146.57.248.225:22 10A6CD36A537FCE513A322361547444B393989F0 cert=K1gDtDAIcUfeLqbstggjIw2rtgIKqdIhUlHp82XRqNSq/mtAjp1BIC9vHKJ2FAEpGssTPw iat-mode=0 Bridge obfs4 146.57.248.225:22 10A6CD36A537FCE513A322361547444B393989F0 cert=K1gDtDAIcUfeLqbstggjIw2rtgIKqdIhUlHp82XRqNSq/mtAjp1BIC9vHKJ2FAEpGssTPw iat-mode=0
Bridge obfs4 45.145.95.6:27015 C5B7CD6946FF10C5B3E89691A7D3F2C122D2117C cert=TD7PbUO0/0k6xYHMPW3vJxICfkMZNdkRrb63Zhl5j9dW3iRGiCx0A7mPhe5T2EDzQ35+Zw iat-mode=0 Bridge obfs4 85.31.186.26:443 91A6354697E6B02A386312F68D82CF86824D3606 cert=PBwr+S8JTVZo6MPdHnkTwXJPILWADLqfMGoVvhZClMq/Urndyd42BwX9YFJHZnBB3H0XCw iat-mode=0
Bridge obfs4 [2a0c:4d80:42:702::1]:27015 C5B7CD6946FF10C5B3E89691A7D3F2C122D2117C cert=TD7PbUO0/0k6xYHMPW3vJxICfkMZNdkRrb63Zhl5j9dW3iRGiCx0A7mPhe5T2EDzQ35+Zw iat-mode=0
Bridge obfs4 51.222.13.177:80 5EDAC3B810E12B01F6FD8050D2FD3E277B289A08 cert=2uplIpLQ0q9+0qMFrK5pkaYRDOe460LL9WHBvatgkuRr/SL31wBOEupaMMJ6koRE6Ld0ew iat-mode=0 Bridge obfs4 51.222.13.177:80 5EDAC3B810E12B01F6FD8050D2FD3E277B289A08 cert=2uplIpLQ0q9+0qMFrK5pkaYRDOe460LL9WHBvatgkuRr/SL31wBOEupaMMJ6koRE6Ld0ew iat-mode=0
UseBridges 1 Bridge obfs4 209.148.46.65:443 74FAD13168806246602538555B5521A0383A1875 cert=ssH+9rP8dG2NLDN2XuFw63hIO/9MNNinLmxQDpVa+7kTOa9/m+tGWT1SmSYpQ9uTBGa6Hw iat-mode=0
Bridge obfs4 45.145.95.6:27015 C5B7CD6946FF10C5B3E89691A7D3F2C122D2117C cert=TD7PbUO0/0k6xYHMPW3vJxICfkMZNdkRrb63Zhl5j9dW3iRGiCx0A7mPhe5T2EDzQ35+Zw iat-mode=0
Bridge obfs4 38.229.1.78:80 C8CBDB2464FC9804A69531437BCF2BE31FDD2EE4 cert=Hmyfd2ev46gGY7NoVxA9ngrPF2zCZtzskRTzoWXbxNkzeVnGFPWmrTtILRyqCTjHR+s9dg iat-mode=1
Bridge obfs4 193.11.166.194:27025 1AE2C08904527FEA90C4C4F8C1083EA59FBC6FAF cert=ItvYZzW5tn6v3G4UnQa6Qz04Npro6e81AP70YujmK/KXwDFPTs3aHXcHp4n8Vt6w/bv8cA iat-mode=0
Bridge obfs4 193.11.166.194:27020 86AC7B8D430DAC4117E9F42C9EAED18133863AAF cert=0LDeJH4JzMDtkJJrFphJCiPqKx7loozKN7VNfuukMGfHO0Z8OGdzHVkhVAOfo1mUdv9cMg iat-mode=0
Bridge obfs4 37.218.245.14:38224 D9A82D2F9C2F65A18407B1D2B764F130847F8B5D cert=bjRaMrr1BRiAW8IE9U5z27fQaYgOhX1UCmOpg2pFpoMvo6ZgQMzLsaTzzQNTlm7hNcb+Sg iat-mode=0
Bridge obfs4 144.217.20.138:80 FB70B257C162BF1038CA669D568D76F5B7F0BABB cert=vYIV5MgrghGQvZPIi1tJwnzorMgqgmlKaB77Y3Z9Q/v94wZBOAXkW+fdx4aSxLVnKO+xNw iat-mode=0
Bridge obfs4 192.95.36.142:443 CDF2E852BF539B82BD10E27E9115A31734E378C2 cert=qUVQ0srL1JI/vO6V6m/24anYXiJD3QP2HgzUKQtQ7GRqqUvs7P+tG43RtAqdhLOALP7DJQ iat-mode=1
Bridge obfs4 [2a0c:4d80:42:702::1]:27015 C5B7CD6946FF10C5B3E89691A7D3F2C122D2117C cert=TD7PbUO0/0k6xYHMPW3vJxICfkMZNdkRrb63Zhl5j9dW3iRGiCx0A7mPhe5T2EDzQ35+Zw iat-mode=0

View file

@ -1,3 +1 @@
# Enable built-in snowflake bridge Bridge snowflake 0.0.3.0:1 2B280B23E1107BB62ABFC40DDCC8824814F80A72
Bridge snowflake 192.0.2.3:1 2B280B23E1107BB62ABFC40DDCC8824814F80A72
UseBridges 1

View file

@ -110,6 +110,7 @@ class Settings(object):
"bridges_builtin_pt": "obfs4", # "obfs4", "meek-azure", or "snowflake" "bridges_builtin_pt": "obfs4", # "obfs4", "meek-azure", or "snowflake"
"bridges_moat": "", "bridges_moat": "",
"bridges_custom": "", "bridges_custom": "",
"bridges_builtin": {},
"persistent_tabs": [], "persistent_tabs": [],
"locale": None, # this gets defined in fill_in_defaults() "locale": None, # this gets defined in fill_in_defaults()
"theme": 0, "theme": 0,

View file

@ -34,6 +34,7 @@ class TestSettings:
"bridges_builtin_pt": "obfs4", "bridges_builtin_pt": "obfs4",
"bridges_moat": "", "bridges_moat": "",
"bridges_custom": "", "bridges_custom": "",
"bridges_builtin": {},
"persistent_tabs": [], "persistent_tabs": [],
"theme": 0, "theme": 0,
} }

View file

@ -507,5 +507,4 @@ class GuiCommon:
return strings._("error_stealth_not_supported") return strings._("error_stealth_not_supported")
elif type(e) is PortNotAvailable: elif type(e) is PortNotAvailable:
return strings._("error_port_not_available") return strings._("error_port_not_available")
return None return None