aboutsummaryrefslogtreecommitdiff
path: root/desktop/scripts/bridges/__init__.py
blob: e2495e3875d9c03cde04c637377a80ed9bd49a5b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
OnionShare | https://onionshare.org/

Copyright (C) 2014-2022 Micah Lee, et al. <micah@micahflee.com>

This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program.  If not, see <http://www.gnu.org/licenses/>.
"""

import os
import requests


class UpdateTorBridges:
    """
    Update the built-in Tor Bridges in OnionShare's torrc templates.
    """

    def __init__(self, root_path):
        self.root_path = root_path
        torrc_template_dir = os.path.join(
            self.root_path, os.pardir, "cli/onionshare_cli/resources"
        )
        endpoint = "https://bridges.torproject.org/moat/circumvention/builtin"
        r = requests.post(
            endpoint,
            headers={"Content-Type": "application/vnd.api+json"},
        )
        if r.status_code != 200:
            print(
                f"There was a problem fetching the latest built-in bridges: status_code={r.status_code}"
            )
            return False

        result = r.json()

        if "errors" in result:
            print(
                f"There was a problem fetching the latest built-in bridges: errors={result['errors']}"
            )
            return False

        for bridge_type in ["meek", "obfs4", "snowflake"]:
            if result[bridge_type]:
                if bridge_type == "meek":
                    torrc_template_extension = "meek_lite_azure"
                else:
                    torrc_template_extension = bridge_type
                torrc_template = os.path.join(
                    self.root_path,
                    torrc_template_dir,
                    f"torrc_template-{torrc_template_extension}",
                )

                with open(torrc_template, "w") as f:
                    f.write(f"# Enable built-in {bridge_type} bridge\n")
                    bridges = result[bridge_type]
                    # Sorts the bridges numerically by IP, since they come back in
                    # random order from the API each time, and create noisy git diff.
                    bridges.sort(key=lambda s: s.split()[1])
                    for item in bridges:
                        if bridge_type == "meek":
                            # obfs4proxy expects the bridge type to be meek_lite, and the url/front params
                            # are missing in the Tor API response, so we have to add them in ourselves.
                            bridge = item.replace("meek", "meek_lite")
                            f.write(
                                f"Bridge {bridge} url=https://meek.azureedge.net/ front=ajax.aspnetcdn.com\n"
                            )
                        else:
                            f.write(f"Bridge {item}\n")