2021-04-27 16:36:57 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
""" Build script for ooniprobe. You can get documentation regarding
|
|
|
|
its usage by running `./make --help`. """
|
|
|
|
|
|
|
|
from __future__ import annotations
|
|
|
|
import datetime
|
|
|
|
|
|
|
|
import getopt
|
|
|
|
import json
|
|
|
|
import os
|
|
|
|
import platform
|
|
|
|
import shlex
|
|
|
|
import shutil
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
|
|
|
|
from typing import Dict
|
|
|
|
from typing import List
|
|
|
|
from typing import NoReturn
|
|
|
|
from typing import Optional
|
|
|
|
from typing import Protocol
|
|
|
|
|
|
|
|
|
|
|
|
def android_cmdlinetools_os() -> str:
|
|
|
|
"""android_cmdlinetools_os maps the name of the current OS to the
|
|
|
|
name used by the android command-line tools file."""
|
|
|
|
system = platform.system()
|
|
|
|
if system == "Linux":
|
|
|
|
return "linux"
|
|
|
|
if system == "Darwin":
|
|
|
|
return "mac"
|
|
|
|
raise RuntimeError(system)
|
|
|
|
|
|
|
|
|
|
|
|
def android_cmdlinetools_version() -> str:
|
|
|
|
"""android_cmdlinetools_version returns the version of the Android
|
|
|
|
command-line tools that we'd like to download."""
|
|
|
|
return "6858069"
|
|
|
|
|
|
|
|
|
|
|
|
def android_cmdlinetools_sha256sum() -> str:
|
|
|
|
"""android_cmdlinetools_sha256sum returns the SHA256 sum of the
|
|
|
|
Android command-line tools zip file."""
|
|
|
|
return {
|
|
|
|
"linux": "87f6dcf41d4e642e37ba03cb2e387a542aa0bd73cb689a9e7152aad40a6e7a08",
|
|
|
|
"mac": "58a55d9c5bcacd7c42170d2cf2c9ae2889c6797a6128307aaf69100636f54a13",
|
|
|
|
}[android_cmdlinetools_os()]
|
|
|
|
|
|
|
|
|
|
|
|
def cachedir() -> str:
|
|
|
|
"""cachedir returns the directory where we cache the SDKs."""
|
|
|
|
return os.path.join(os.path.expandvars("${HOME}"), ".ooniprobe-build")
|
|
|
|
|
|
|
|
|
|
|
|
def goversion() -> str:
|
|
|
|
"""goversion is the Go version we use."""
|
|
|
|
return "1.16.3"
|
|
|
|
|
|
|
|
|
|
|
|
def gopath() -> str:
|
|
|
|
"""gopath is the GOPATH we use."""
|
|
|
|
return os.path.expandvars("${HOME}/go")
|
|
|
|
|
|
|
|
|
|
|
|
def gosha256sum() -> str:
|
|
|
|
"""gosha256sum returns the SHA256 sum of the Go tarball."""
|
|
|
|
return {
|
|
|
|
"linux": {
|
|
|
|
"amd64": "951a3c7c6ce4e56ad883f97d9db74d3d6d80d5fec77455c6ada6c1f7ac4776d2",
|
|
|
|
"arm64": "566b1d6f17d2bc4ad5f81486f0df44f3088c3ed47a3bec4099d8ed9939e90d5d",
|
|
|
|
},
|
|
|
|
"darwin": {
|
|
|
|
"amd64": "6bb1cf421f8abc2a9a4e39140b7397cdae6aca3e8d36dcff39a1a77f4f1170ac",
|
|
|
|
"arm64": "f4e96bbcd5d2d1942f5b55d9e4ab19564da4fad192012f6d7b0b9b055ba4208f",
|
|
|
|
},
|
|
|
|
}[goos()][goarch()]
|
|
|
|
|
|
|
|
|
|
|
|
def goos() -> str:
|
|
|
|
"""goos returns the GOOS value for the current system."""
|
|
|
|
system = platform.system()
|
|
|
|
if system == "Linux":
|
|
|
|
return "linux"
|
|
|
|
if system == "Darwin":
|
|
|
|
return "darwin"
|
|
|
|
raise RuntimeError(system)
|
|
|
|
|
|
|
|
|
|
|
|
def goarch() -> str:
|
|
|
|
"""goarch returns the GOARCH value for the current system."""
|
|
|
|
machine = platform.machine()
|
|
|
|
if machine in ("arm64", "arm", "386", "amd64"):
|
|
|
|
return machine
|
|
|
|
if machine in ("x86", "i386"):
|
|
|
|
return "386"
|
|
|
|
if machine == "x86_64":
|
|
|
|
return "amd64"
|
|
|
|
if machine == "aarch64":
|
|
|
|
return "arm64"
|
|
|
|
raise RuntimeError(machine)
|
|
|
|
|
|
|
|
|
|
|
|
def android_ndk_version() -> str:
|
|
|
|
"""android_ndk_version returns the Android NDK version."""
|
|
|
|
return "22.1.7171670"
|
|
|
|
|
|
|
|
|
|
|
|
def sdkmanager_install_cmd(binpath: str) -> List[str]:
|
|
|
|
"""sdkmanager_install_cmd returns the command line for installing
|
|
|
|
all the required dependencies using the sdkmanager."""
|
|
|
|
return [
|
|
|
|
os.path.join(binpath, "sdkmanager"),
|
|
|
|
"--install",
|
|
|
|
"build-tools;29.0.3",
|
|
|
|
"platforms;android-30",
|
|
|
|
"ndk;{}".format(android_ndk_version()),
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
def log(msg: str) -> None:
|
|
|
|
"""log prints a message on the standard error."""
|
|
|
|
print(msg, file=sys.stderr)
|
|
|
|
|
|
|
|
|
|
|
|
class Options(Protocol):
|
|
|
|
"""Options contains the configured options."""
|
|
|
|
|
|
|
|
def debugging(self) -> bool:
|
|
|
|
"""debugging indicates whether to pass -x to `go build...`."""
|
|
|
|
|
2021-04-29 10:30:39 +02:00
|
|
|
def disable_embedding_psiphon_config(self) -> bool:
|
|
|
|
"""disable_embedding_psiphon_config indicates that the user
|
|
|
|
does not want us to embed an encrypted psiphon config file into
|
|
|
|
the binary."""
|
|
|
|
|
2021-04-27 16:36:57 +02:00
|
|
|
def dry_run(self) -> bool:
|
|
|
|
"""dry_run indicates whether to execute commands."""
|
|
|
|
|
|
|
|
def target(self) -> str:
|
|
|
|
"""target is the target to build."""
|
|
|
|
|
|
|
|
def verbose(self) -> bool:
|
|
|
|
"""verbose indicates whether to pass -v to `go build...`."""
|
|
|
|
|
|
|
|
|
|
|
|
class ConfigFromCLI:
|
|
|
|
"""ConfigFromCLI parses options from CLI flags."""
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def parse(cls, targets: List[str]) -> ConfigFromCLI:
|
|
|
|
"""parse parses command line options and returns a
|
|
|
|
suitable configuration object."""
|
|
|
|
conf = cls()
|
|
|
|
conf._parse(targets)
|
|
|
|
return conf
|
|
|
|
|
|
|
|
def __init__(self) -> None:
|
|
|
|
self._debugging = False
|
2021-04-29 10:30:39 +02:00
|
|
|
self._disable_embedding_psiphon_config = False
|
2021-04-27 16:36:57 +02:00
|
|
|
self._dry_run = False
|
|
|
|
self._target = ""
|
|
|
|
self._verbose = False
|
|
|
|
|
|
|
|
def debugging(self) -> bool:
|
|
|
|
return self._debugging
|
|
|
|
|
2021-04-29 10:30:39 +02:00
|
|
|
def disable_embedding_psiphon_config(self) -> bool:
|
|
|
|
return self._disable_embedding_psiphon_config
|
|
|
|
|
2021-04-27 16:36:57 +02:00
|
|
|
def dry_run(self) -> bool:
|
|
|
|
return self._dry_run
|
|
|
|
|
|
|
|
def target(self) -> str:
|
|
|
|
return self._target
|
|
|
|
|
|
|
|
def verbose(self) -> bool:
|
|
|
|
return self._verbose
|
|
|
|
|
|
|
|
# The main reason why I am using getopt here is such that I am able
|
|
|
|
# to print a very clear and detailed usage string. (But the same
|
|
|
|
# could be obtained quite likely w/ argparse.)
|
|
|
|
|
|
|
|
__usage_string = """\
|
2021-04-29 10:30:39 +02:00
|
|
|
usage: ./make [--disable-embedding-psiphon-config] [-nvx] -t target
|
2021-04-27 16:36:57 +02:00
|
|
|
./make -l
|
|
|
|
./make [--help|-h]
|
|
|
|
|
|
|
|
The first form of the command builds the `target` specified using the
|
|
|
|
`-t` command line flag. If the target has dependencies, this command will
|
|
|
|
build the dependent targets first. The `-n` flag enables a dry run where
|
|
|
|
the command only prints the commands it would run. The `-v` and `-x` flags
|
2021-04-29 10:30:39 +02:00
|
|
|
are passed directly to `go build ...` and `gomobile bind ...`. The
|
|
|
|
`--disable-embedding-psiphon-config` flag causes this command to disable
|
|
|
|
embedding a psiphon config file into the generated binary; you should
|
|
|
|
use this option when you cannot clone the private repository containing
|
|
|
|
the psiphon configuration file.
|
2021-04-27 16:36:57 +02:00
|
|
|
|
|
|
|
The second form of the command lists all the available targets as
|
|
|
|
a pretty-printed JSON list.
|
|
|
|
|
|
|
|
The third form of the command prints this help screen.
|
|
|
|
"""
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def _usage(cls, err: str = "", exitcode: int = 0) -> NoReturn:
|
|
|
|
if err:
|
|
|
|
sys.stderr.write("error: {}\n".format(err))
|
|
|
|
sys.stderr.write(cls.__usage_string)
|
|
|
|
sys.exit(exitcode)
|
|
|
|
|
|
|
|
def _parse(self, targets: List[str]):
|
|
|
|
try:
|
2021-04-29 10:30:39 +02:00
|
|
|
opts, args = getopt.getopt(
|
|
|
|
sys.argv[1:], "hlnt:vx", ["disable-embedding-psiphon-config", "help"]
|
|
|
|
)
|
2021-04-27 16:36:57 +02:00
|
|
|
except getopt.GetoptError as err:
|
|
|
|
self._usage(err=err.msg, exitcode=1)
|
|
|
|
if args:
|
|
|
|
self._usage(err="unexpected number of positional arguments", exitcode=1)
|
|
|
|
for key, value in opts:
|
2021-04-29 10:30:39 +02:00
|
|
|
if key == "--disable-embedding-psiphon-config":
|
|
|
|
self._disable_embedding_psiphon_config = True
|
|
|
|
continue
|
2021-04-27 16:36:57 +02:00
|
|
|
if key in ("-h", "--help"):
|
|
|
|
self._usage()
|
|
|
|
if key == "-l":
|
|
|
|
sys.stdout.write("{}\n".format(json.dumps(targets, indent=4)))
|
|
|
|
sys.exit(0)
|
|
|
|
if key == "-n":
|
|
|
|
self._dry_run = True
|
|
|
|
continue
|
|
|
|
if key == "-t":
|
|
|
|
self._target = value
|
|
|
|
continue
|
|
|
|
if key == "-v":
|
|
|
|
self._verbose = True
|
|
|
|
continue
|
|
|
|
if key == "-x":
|
|
|
|
self._debugging = True
|
|
|
|
continue
|
|
|
|
raise RuntimeError(key, value)
|
|
|
|
|
|
|
|
if self._target == "": # no arguments is equivalent to --help
|
|
|
|
self._usage()
|
|
|
|
|
|
|
|
if self._target not in targets:
|
|
|
|
sys.stderr.write("unknown target: {}\n".format(self._target))
|
|
|
|
sys.stderr.write("try `./make -l` to see the available targets.\n")
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
|
|
|
|
class Engine(Protocol):
|
|
|
|
"""Engine is an engine for building targets."""
|
|
|
|
|
|
|
|
def cat_sed_redirect(
|
|
|
|
self, pattern: str, value: str, source: str, dest: str
|
|
|
|
) -> None:
|
|
|
|
"""cat_sed_redirect does `cat $source|sed "s/$pattern/$value/g" > $dest`."""
|
|
|
|
|
|
|
|
def echo_to_file(self, content: str, filepath: str) -> None:
|
|
|
|
"""echo_to_file writes the content string to the given file."""
|
|
|
|
|
|
|
|
def require(self, *executable: str) -> None:
|
|
|
|
"""require fails if executable is not found in path."""
|
|
|
|
|
|
|
|
def run(
|
|
|
|
self,
|
|
|
|
cmdline: List[str],
|
|
|
|
cwd: Optional[str] = None,
|
|
|
|
extra_env: Optional[Dict[str, str]] = None,
|
|
|
|
inputbytes: Optional[bytes] = None,
|
|
|
|
) -> None:
|
|
|
|
"""run runs the specified command line."""
|
|
|
|
|
|
|
|
|
|
|
|
class CommandRealExecutor:
|
|
|
|
"""CommandRealExecutor executes commands."""
|
|
|
|
|
|
|
|
def cat_sed_redirect(
|
|
|
|
self, pattern: str, value: str, source: str, dest: str
|
|
|
|
) -> None:
|
|
|
|
"""cat_sed_redirect implements Engine.cat_sed_redirect."""
|
|
|
|
with open(source, "r") as sourcefp:
|
|
|
|
data = sourcefp.read().replace(pattern, value)
|
|
|
|
with open(dest, "w") as destfp:
|
|
|
|
destfp.write(data)
|
|
|
|
|
|
|
|
def echo_to_file(self, content: str, filepath: str) -> None:
|
|
|
|
"""echo_to_file implements Engine.echo_to_file"""
|
|
|
|
with open(filepath, "w") as filep:
|
|
|
|
filep.write(content)
|
|
|
|
filep.write("\n")
|
|
|
|
|
|
|
|
def require(self, *executable: str) -> None:
|
|
|
|
"""require implements Engine.require."""
|
|
|
|
# Implemented in CommandDryRunner
|
|
|
|
|
|
|
|
def run(
|
|
|
|
self,
|
|
|
|
cmdline: List[str],
|
|
|
|
cwd: Optional[str] = None,
|
|
|
|
extra_env: Optional[Dict[str, str]] = None,
|
|
|
|
inputbytes: Optional[bytes] = None,
|
|
|
|
) -> None:
|
|
|
|
"""run implements Engine.run."""
|
|
|
|
env = os.environ.copy()
|
|
|
|
if extra_env:
|
|
|
|
for key, value in extra_env.items():
|
|
|
|
env[key] = value
|
|
|
|
subprocess.run(cmdline, check=True, cwd=cwd, env=env, input=inputbytes)
|
|
|
|
|
|
|
|
|
|
|
|
class CommandDryRunner:
|
|
|
|
"""CommandDryRunner is the dry runner."""
|
|
|
|
|
|
|
|
# Implementation note: here we try to log valid bash snippets
|
|
|
|
# such that is really obvious what we are doing.
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self.__cmdcache: Dict[str, str] = {}
|
|
|
|
|
|
|
|
def cat_sed_redirect(
|
|
|
|
self, pattern: str, value: str, source: str, dest: str
|
|
|
|
) -> None:
|
|
|
|
"""cat_sed_redirect implements Engine.cat_sed_redirect."""
|
|
|
|
log('./make: cat {}|sed "s/{}/{}/g" > {}'.format(source, pattern, value, dest))
|
|
|
|
|
|
|
|
def echo_to_file(self, content: str, filepath: str) -> None:
|
|
|
|
"""echo_to_file implements Engine.echo_to_file"""
|
|
|
|
log("./make: echo '{}' > {}".format(content, filepath))
|
|
|
|
|
|
|
|
def require(self, *executable: str) -> None:
|
|
|
|
"""require implements Engine.require."""
|
|
|
|
for exc in executable:
|
|
|
|
if exc in self.__cmdcache:
|
|
|
|
continue # do not print checks more than once
|
|
|
|
fullpath = shutil.which(exc)
|
|
|
|
if not fullpath:
|
|
|
|
log("./make: checking for {}... not found".format(exc))
|
|
|
|
sys.exit(1)
|
|
|
|
log("./make: checking for {}... {}".format(exc, fullpath))
|
|
|
|
self.__cmdcache[exc] = fullpath
|
|
|
|
|
|
|
|
def run(
|
|
|
|
self,
|
|
|
|
cmdline: List[str],
|
|
|
|
cwd: Optional[str] = None,
|
|
|
|
extra_env: Optional[Dict[str, str]] = None,
|
|
|
|
inputbytes: Optional[bytes] = None,
|
|
|
|
) -> None:
|
|
|
|
"""run implements Engine.run."""
|
|
|
|
cdpart = ""
|
|
|
|
if cwd:
|
|
|
|
cdpart = "cd {} && ".format(cwd)
|
|
|
|
envpart = ""
|
|
|
|
if extra_env:
|
|
|
|
for key, value in extra_env.items():
|
|
|
|
envpart += shlex.join(["{}={}".format(key, value)]) + " "
|
|
|
|
log("./make: {}{}{}".format(cdpart, envpart, shlex.join(cmdline)))
|
|
|
|
|
|
|
|
|
|
|
|
class EngineComposer:
|
|
|
|
"""EngineComposer composes two engines."""
|
|
|
|
|
|
|
|
def __init__(self, first: Engine, second: Engine):
|
|
|
|
self._first = first
|
|
|
|
self._second = second
|
|
|
|
|
|
|
|
def cat_sed_redirect(
|
|
|
|
self, pattern: str, value: str, source: str, dest: str
|
|
|
|
) -> None:
|
|
|
|
"""cat_sed_redirect implements Engine.cat_sed_redirect."""
|
|
|
|
self._first.cat_sed_redirect(pattern, value, source, dest)
|
|
|
|
self._second.cat_sed_redirect(pattern, value, source, dest)
|
|
|
|
|
|
|
|
def echo_to_file(self, content: str, filepath: str) -> None:
|
|
|
|
"""echo_to_file implements Engine.echo_to_file"""
|
|
|
|
self._first.echo_to_file(content, filepath)
|
|
|
|
self._second.echo_to_file(content, filepath)
|
|
|
|
|
|
|
|
def require(self, *executable: str) -> None:
|
|
|
|
"""require implements Engine.require."""
|
|
|
|
self._first.require(*executable)
|
|
|
|
self._second.require(*executable)
|
|
|
|
|
|
|
|
def run(
|
|
|
|
self,
|
|
|
|
cmdline: List[str],
|
|
|
|
cwd: Optional[str] = None,
|
|
|
|
extra_env: Optional[Dict[str, str]] = None,
|
|
|
|
inputbytes: Optional[bytes] = None,
|
|
|
|
) -> None:
|
|
|
|
"""run implements Engine.run."""
|
|
|
|
self._first.run(cmdline, cwd=cwd, extra_env=extra_env, inputbytes=inputbytes)
|
|
|
|
self._second.run(cmdline, cwd=cwd, extra_env=extra_env, inputbytes=inputbytes)
|
|
|
|
|
|
|
|
|
|
|
|
def new_engine(options: Options) -> Engine:
|
|
|
|
"""new_engine creates a new engine instance"""
|
|
|
|
if options.dry_run():
|
|
|
|
return CommandDryRunner()
|
|
|
|
return EngineComposer(CommandDryRunner(), CommandRealExecutor())
|
|
|
|
|
|
|
|
|
|
|
|
class Target(Protocol):
|
|
|
|
"""Target is a target to build."""
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
"""name returns the target name."""
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
"""build builds the specified target."""
|
|
|
|
|
|
|
|
|
|
|
|
class SDKGolangGo:
|
|
|
|
"""SDKGolangGo creates ${cachedir}/SDK/golang."""
|
|
|
|
|
|
|
|
# We download a golang SDK from upstream to make sure we
|
|
|
|
# are always using a specific version of golang/go.
|
|
|
|
|
|
|
|
__name = os.path.join(cachedir(), "SDK", "golang")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
2021-04-29 10:30:39 +02:00
|
|
|
def binpath(self) -> str:
|
|
|
|
return os.path.join(self.__name, "go", "bin")
|
|
|
|
|
2021-04-27 16:36:57 +02:00
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isdir(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
|
|
|
log("./make: building {}...".format(self.__name))
|
|
|
|
engine.require("mkdir", "curl", "shasum", "rm", "tar", "echo")
|
|
|
|
filename = "go{}.{}-{}.tar.gz".format(goversion(), goos(), goarch())
|
|
|
|
url = "https://golang.org/dl/{}".format(filename)
|
|
|
|
engine.run(["mkdir", "-p", self.__name])
|
|
|
|
filepath = os.path.join(self.__name, filename)
|
|
|
|
engine.run(["curl", "-fsSLo", filepath, url])
|
|
|
|
sha256file = os.path.join(cachedir(), "SDK", "SHA256")
|
|
|
|
engine.echo_to_file("{} {}".format(gosha256sum(), filepath), sha256file)
|
|
|
|
engine.run(["shasum", "--check", sha256file])
|
|
|
|
engine.run(["rm", sha256file])
|
|
|
|
engine.run(["tar", "-xf", filename], cwd=self.__name)
|
|
|
|
engine.run(["rm", filepath])
|
|
|
|
|
|
|
|
def goroot(self):
|
|
|
|
"""goroot returns the goroot."""
|
|
|
|
return os.path.join(self.__name, "go")
|
|
|
|
|
|
|
|
|
|
|
|
class SDKOONIGo:
|
|
|
|
"""SDKOONIGo creates ${cachedir}/SDK/oonigo."""
|
|
|
|
|
|
|
|
# We use a private fork of golang/go on Android as a
|
|
|
|
# workaround for https://github.com/ooni/probe/issues/1444
|
|
|
|
|
|
|
|
__name = os.path.join(cachedir(), "SDK", "oonigo")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def binpath(self) -> str:
|
|
|
|
return os.path.join(self.__name, "bin")
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isdir(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
|
|
|
golang_go = SDKGolangGo()
|
|
|
|
golang_go.build(engine, options)
|
|
|
|
log("./make: building {}...".format(self.__name))
|
|
|
|
engine.require("git", "bash")
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"git",
|
|
|
|
"clone",
|
|
|
|
"-b",
|
|
|
|
"ooni",
|
|
|
|
"--single-branch",
|
|
|
|
"--depth",
|
|
|
|
"8",
|
|
|
|
"https://github.com/ooni/go",
|
|
|
|
self.__name,
|
|
|
|
]
|
|
|
|
)
|
|
|
|
engine.run(
|
|
|
|
["./make.bash"],
|
|
|
|
cwd=os.path.join(self.__name, "src"),
|
|
|
|
extra_env={"GOROOT_BOOTSTRAP": golang_go.goroot()},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class SDKAndroid:
|
|
|
|
"""SDKAndroid creates ${cachedir}/SDK/android."""
|
|
|
|
|
|
|
|
__name = os.path.join(cachedir(), "SDK", "android")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def home(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def ndk_home(self) -> str:
|
|
|
|
return os.path.join(self.home(), "ndk", android_ndk_version())
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isdir(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
|
|
|
log("./make: building {}...".format(self.__name))
|
|
|
|
engine.require("mkdir", "curl", "echo", "shasum", "rm", "unzip", "mv", "java")
|
|
|
|
filename = "commandlinetools-{}-{}_latest.zip".format(
|
|
|
|
android_cmdlinetools_os(), android_cmdlinetools_version()
|
|
|
|
)
|
|
|
|
url = "https://dl.google.com/android/repository/{}".format(filename)
|
|
|
|
engine.run(["mkdir", "-p", self.__name])
|
|
|
|
filepath = os.path.join(self.__name, filename)
|
|
|
|
engine.run(["curl", "-fsSLo", filepath, url])
|
|
|
|
sha256file = os.path.join(cachedir(), "SDK", "SHA256")
|
|
|
|
engine.echo_to_file(
|
|
|
|
"{} {}".format(android_cmdlinetools_sha256sum(), filepath), sha256file
|
|
|
|
)
|
|
|
|
engine.run(["shasum", "--check", sha256file])
|
|
|
|
engine.run(["rm", sha256file])
|
|
|
|
engine.run(["unzip", filename], cwd=self.__name)
|
|
|
|
engine.run(["rm", filepath])
|
|
|
|
# See https://stackoverflow.com/a/61176718 to understand why
|
|
|
|
# we need to reorganize the directories like this:
|
|
|
|
engine.run(
|
|
|
|
["mv", "cmdline-tools", android_cmdlinetools_version()], cwd=self.__name
|
|
|
|
)
|
|
|
|
engine.run(["mkdir", "cmdline-tools"], cwd=self.__name)
|
|
|
|
engine.run(
|
|
|
|
["mv", android_cmdlinetools_version(), "cmdline-tools"], cwd=self.__name
|
|
|
|
)
|
|
|
|
engine.run(
|
|
|
|
sdkmanager_install_cmd(
|
|
|
|
os.path.join(
|
|
|
|
self.__name,
|
|
|
|
"cmdline-tools",
|
|
|
|
android_cmdlinetools_version(),
|
|
|
|
"bin",
|
|
|
|
),
|
|
|
|
),
|
|
|
|
inputbytes=b"Y\n", # automatically accept license
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class OONIProbePrivate:
|
|
|
|
"""OONIProbePrivate creates ${cachedir}/github.com/ooni/probe-private."""
|
|
|
|
|
|
|
|
# We use this private repository to copy the psiphon configuration
|
|
|
|
# file to embed into the ooniprobe binaries
|
|
|
|
|
|
|
|
__name = os.path.join(cachedir(), "github.com", "ooni", "probe-private")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
2021-04-29 10:30:39 +02:00
|
|
|
def copyfiles(self, engine: Engine, options: Options) -> None:
|
2021-04-27 16:36:57 +02:00
|
|
|
"""copyfiles copies psiphon config to the repository."""
|
2021-04-29 10:30:39 +02:00
|
|
|
if options.disable_embedding_psiphon_config():
|
|
|
|
log("./make: copy psiphon config: disabled by command line flags")
|
|
|
|
return
|
2021-04-27 16:36:57 +02:00
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"cp",
|
|
|
|
os.path.join(self.__name, "psiphon-config.json.age"),
|
|
|
|
os.path.join("internal", "engine"),
|
|
|
|
]
|
|
|
|
)
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"cp",
|
|
|
|
os.path.join(self.__name, "psiphon-config.key"),
|
|
|
|
os.path.join("internal", "engine"),
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isdir(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
2021-04-29 10:30:39 +02:00
|
|
|
if options.disable_embedding_psiphon_config():
|
|
|
|
log("./make: {}: disabled by command line flags".format(self.__name))
|
|
|
|
return
|
2021-04-27 16:36:57 +02:00
|
|
|
log("./make: building {}...".format(self.__name))
|
|
|
|
engine.require("git", "cp")
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"git",
|
|
|
|
"clone",
|
|
|
|
"git@github.com:ooni/probe-private",
|
|
|
|
self.__name,
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class OONIMKAllAAR:
|
|
|
|
"""OONIMKAllAAR creates ./MOBILE/android/oonimkall.aar."""
|
|
|
|
|
|
|
|
__name = os.path.join(".", "MOBILE", "android", "oonimkall.aar")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def aarfile(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def srcfile(self) -> str:
|
|
|
|
return os.path.join(".", "MOBILE", "android", "oonimkall-sources.jar")
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isfile(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
|
|
|
ooprivate = OONIProbePrivate()
|
|
|
|
ooprivate.build(engine, options)
|
|
|
|
oonigo = SDKOONIGo()
|
|
|
|
oonigo.build(engine, options)
|
|
|
|
android = SDKAndroid()
|
|
|
|
android.build(engine, options)
|
|
|
|
log("./make: building {}...".format(self.__name))
|
2021-04-29 10:30:39 +02:00
|
|
|
ooprivate.copyfiles(engine, options)
|
2021-04-27 16:36:57 +02:00
|
|
|
engine.require("sh", "javac")
|
|
|
|
self._go_get_gomobile(engine, options, oonigo)
|
|
|
|
self._gomobile_init(engine, oonigo, android)
|
|
|
|
self._gomobile_bind(engine, options, oonigo, android)
|
|
|
|
|
|
|
|
# Implementation note: we use proxy scripts for go and gomobile
|
|
|
|
# that explicitly print what they resolve go and gomobile to using
|
|
|
|
# `command -v`. This gives us extra confidence that we are really
|
|
|
|
# using the oonigo fork of golang/go.
|
|
|
|
|
|
|
|
def _go_get_gomobile(
|
|
|
|
self, engine: Engine, options: Options, oonigo: SDKOONIGo
|
|
|
|
) -> None:
|
|
|
|
# TODO(bassosimone): find a way to run this command without
|
|
|
|
# adding extra dependencies to go.mod and go.sum.
|
|
|
|
cmdline: List[str] = []
|
2021-04-29 10:30:39 +02:00
|
|
|
cmdline.append(os.path.join(".", "MOBILE", "go"))
|
2021-04-27 16:36:57 +02:00
|
|
|
cmdline.append("get")
|
|
|
|
cmdline.append("-u")
|
|
|
|
if options.verbose():
|
|
|
|
cmdline.append("-v")
|
|
|
|
if options.debugging():
|
|
|
|
cmdline.append("-x")
|
|
|
|
cmdline.append("golang.org/x/mobile/cmd/gomobile@latest")
|
|
|
|
engine.run(
|
|
|
|
cmdline,
|
|
|
|
extra_env={
|
|
|
|
"PATH": os.pathsep.join(
|
|
|
|
[
|
|
|
|
oonigo.binpath(), # so we use our go fork
|
|
|
|
os.environ["PATH"], # original path
|
|
|
|
]
|
|
|
|
),
|
|
|
|
"GOPATH": gopath(), # where to install gomobile
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def _gomobile_init(
|
|
|
|
self,
|
|
|
|
engine: Engine,
|
|
|
|
oonigo: SDKOONIGo,
|
|
|
|
android: SDKAndroid,
|
|
|
|
) -> None:
|
|
|
|
cmdline: List[str] = []
|
2021-04-29 10:30:39 +02:00
|
|
|
cmdline.append(os.path.join(".", "MOBILE", "gomobile"))
|
2021-04-27 16:36:57 +02:00
|
|
|
cmdline.append("init")
|
|
|
|
engine.run(
|
|
|
|
cmdline,
|
|
|
|
extra_env={
|
|
|
|
"PATH": os.pathsep.join(
|
|
|
|
[
|
|
|
|
os.path.join(gopath(), "bin"), # for gomobile
|
|
|
|
oonigo.binpath(), # for our go fork
|
|
|
|
os.environ["PATH"], # original environment
|
|
|
|
]
|
|
|
|
),
|
|
|
|
"ANDROID_HOME": android.home(),
|
|
|
|
"ANDROID_NDK_HOME": android.ndk_home(),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def _gomobile_bind(
|
|
|
|
self,
|
|
|
|
engine: Engine,
|
|
|
|
options: Options,
|
|
|
|
oonigo: SDKOONIGo,
|
|
|
|
android: SDKAndroid,
|
|
|
|
) -> None:
|
|
|
|
cmdline: List[str] = []
|
2021-04-29 10:30:39 +02:00
|
|
|
cmdline.append(os.path.join(".", "MOBILE", "gomobile"))
|
2021-04-27 16:36:57 +02:00
|
|
|
cmdline.append("bind")
|
|
|
|
if options.verbose():
|
|
|
|
cmdline.append("-v")
|
|
|
|
if options.debugging():
|
|
|
|
cmdline.append("-x")
|
|
|
|
cmdline.append("-target")
|
|
|
|
cmdline.append("android")
|
|
|
|
cmdline.append("-o")
|
|
|
|
cmdline.append(self.__name)
|
2021-04-29 10:30:39 +02:00
|
|
|
if not options.disable_embedding_psiphon_config():
|
|
|
|
cmdline.append("-tags")
|
|
|
|
cmdline.append("ooni_psiphon_config")
|
2021-04-27 16:36:57 +02:00
|
|
|
cmdline.append("-ldflags")
|
|
|
|
cmdline.append("-s -w")
|
|
|
|
cmdline.append("./pkg/oonimkall")
|
|
|
|
engine.run(
|
|
|
|
cmdline,
|
|
|
|
extra_env={
|
|
|
|
"PATH": os.pathsep.join(
|
|
|
|
[
|
|
|
|
os.path.join(gopath(), "bin"), # for gomobile
|
|
|
|
oonigo.binpath(), # for our go fork
|
|
|
|
os.environ["PATH"], # original environment
|
|
|
|
]
|
|
|
|
),
|
|
|
|
"ANDROID_HOME": android.home(),
|
|
|
|
"ANDROID_NDK_HOME": android.ndk_home(),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class BundleJAR:
|
|
|
|
"""BundleJAR creates ./MOBILE/android/bundle.jar."""
|
|
|
|
|
|
|
|
# We upload the bundle.jar file to maven central to bless
|
|
|
|
# a new release of the OONI libraries for Android.
|
|
|
|
|
|
|
|
__name = os.path.join(".", "MOBILE", "android", "bundle.jar")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isfile(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
|
|
|
oonimkall = OONIMKAllAAR()
|
|
|
|
oonimkall.build(engine, options)
|
|
|
|
log("./make: building {}...".format(self.__name))
|
|
|
|
engine.require("cp", "gpg", "jar")
|
|
|
|
version = datetime.datetime.now().strftime("%Y.%m.%d-%H%M%S")
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"cp",
|
|
|
|
oonimkall.aarfile(),
|
|
|
|
os.path.join("MOBILE", "android", "oonimkall-{}.aar".format(version)),
|
|
|
|
]
|
|
|
|
)
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"cp",
|
|
|
|
oonimkall.srcfile(),
|
|
|
|
os.path.join(
|
|
|
|
"MOBILE", "android", "oonimkall-{}-sources.jar".format(version)
|
|
|
|
),
|
|
|
|
]
|
|
|
|
)
|
|
|
|
engine.cat_sed_redirect(
|
|
|
|
"@VERSION@",
|
|
|
|
version,
|
|
|
|
os.path.join("MOBILE", "template.pom"),
|
|
|
|
os.path.join("MOBILE", "android", "oonimkall-{}.pom".format(version)),
|
|
|
|
)
|
|
|
|
names = (
|
|
|
|
"oonimkall-{}.aar".format(version),
|
|
|
|
"oonimkall-{}-sources.jar".format(version),
|
|
|
|
"oonimkall-{}.pom".format(version),
|
|
|
|
)
|
|
|
|
for name in names:
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"gpg",
|
|
|
|
"-abu",
|
|
|
|
"simone@openobservatory.org",
|
|
|
|
name,
|
|
|
|
],
|
|
|
|
cwd=os.path.join(".", "MOBILE", "android"),
|
|
|
|
)
|
|
|
|
allnames = [name + ".asc" for name in names]
|
|
|
|
allnames.extend(names)
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"jar",
|
|
|
|
"-cf",
|
|
|
|
"bundle.jar",
|
|
|
|
*allnames,
|
|
|
|
],
|
|
|
|
cwd=os.path.join(".", "MOBILE", "android"),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class Android:
|
|
|
|
"""Android is the toplevel android target."""
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return "android"
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
bundlejar = BundleJAR()
|
|
|
|
bundlejar.build(engine, options)
|
|
|
|
|
|
|
|
|
2021-04-29 10:30:39 +02:00
|
|
|
class OONIMKAllFramework:
|
|
|
|
"""OONIMKAllFramework creates ./MOBILE/ios/oonimkall.framework."""
|
|
|
|
|
|
|
|
__name = os.path.join(".", "MOBILE", "ios", "oonimkall.framework")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isfile(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
|
|
|
ooprivate = OONIProbePrivate()
|
|
|
|
ooprivate.build(engine, options)
|
|
|
|
gogo = SDKGolangGo()
|
|
|
|
gogo.build(engine, options)
|
|
|
|
log("./make: building {}...".format(self.__name))
|
|
|
|
ooprivate.copyfiles(engine, options)
|
|
|
|
self._go_get_gomobile(engine, options, gogo)
|
|
|
|
self._gomobile_init(engine, gogo)
|
|
|
|
self._gomobile_bind(engine, options, gogo)
|
|
|
|
|
|
|
|
def _go_get_gomobile(
|
|
|
|
self,
|
|
|
|
engine: Engine,
|
|
|
|
options: Options,
|
|
|
|
gogo: SDKGolangGo,
|
|
|
|
) -> None:
|
|
|
|
# TODO(bassosimone): find a way to run this command without
|
|
|
|
# adding extra dependencies to go.mod and go.sum.
|
|
|
|
cmdline: List[str] = []
|
|
|
|
cmdline.append(os.path.join(".", "MOBILE", "go"))
|
|
|
|
cmdline.append("get")
|
|
|
|
cmdline.append("-u")
|
|
|
|
if options.verbose():
|
|
|
|
cmdline.append("-v")
|
|
|
|
if options.debugging():
|
|
|
|
cmdline.append("-x")
|
|
|
|
cmdline.append("golang.org/x/mobile/cmd/gomobile@latest")
|
|
|
|
engine.run(
|
|
|
|
cmdline,
|
|
|
|
extra_env={
|
|
|
|
"PATH": os.pathsep.join(
|
|
|
|
[
|
|
|
|
gogo.binpath(), # so we use this binary
|
|
|
|
os.environ["PATH"], # original path
|
|
|
|
]
|
|
|
|
),
|
|
|
|
"GOPATH": gopath(), # where to install gomobile
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def _gomobile_init(
|
|
|
|
self,
|
|
|
|
engine: Engine,
|
|
|
|
gogo: SDKGolangGo,
|
|
|
|
) -> None:
|
|
|
|
cmdline: List[str] = []
|
|
|
|
cmdline.append(os.path.join(".", "MOBILE", "gomobile"))
|
|
|
|
cmdline.append("init")
|
|
|
|
engine.run(
|
|
|
|
cmdline,
|
|
|
|
extra_env={
|
|
|
|
"PATH": os.pathsep.join(
|
|
|
|
[
|
|
|
|
os.path.join(gopath(), "bin"), # for gomobile
|
|
|
|
gogo.binpath(), # for our go fork
|
|
|
|
os.environ["PATH"], # original environment
|
|
|
|
]
|
|
|
|
),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
def _gomobile_bind(
|
|
|
|
self,
|
|
|
|
engine: Engine,
|
|
|
|
options: Options,
|
|
|
|
gogo: SDKGolangGo,
|
|
|
|
) -> None:
|
|
|
|
cmdline: List[str] = []
|
|
|
|
cmdline.append(os.path.join(".", "MOBILE", "gomobile"))
|
|
|
|
cmdline.append("bind")
|
|
|
|
if options.verbose():
|
|
|
|
cmdline.append("-v")
|
|
|
|
if options.debugging():
|
|
|
|
cmdline.append("-x")
|
|
|
|
cmdline.append("-target")
|
|
|
|
cmdline.append("ios")
|
|
|
|
cmdline.append("-o")
|
|
|
|
cmdline.append(self.__name)
|
|
|
|
if not options.disable_embedding_psiphon_config():
|
|
|
|
cmdline.append("-tags")
|
|
|
|
cmdline.append("ooni_psiphon_config")
|
|
|
|
cmdline.append("-ldflags")
|
|
|
|
cmdline.append("-s -w")
|
|
|
|
cmdline.append("./pkg/oonimkall")
|
|
|
|
engine.run(
|
|
|
|
cmdline,
|
|
|
|
extra_env={
|
|
|
|
"PATH": os.pathsep.join(
|
|
|
|
[
|
|
|
|
os.path.join(gopath(), "bin"), # for gomobile
|
|
|
|
gogo.binpath(), # for our go fork
|
|
|
|
os.environ["PATH"], # original environment
|
|
|
|
]
|
|
|
|
),
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class OONIMKAllFrameworkZip:
|
|
|
|
"""OONIMKAllFrameworkZip creates ./MOBILE/ios/oonimkall.framework.zip."""
|
|
|
|
|
|
|
|
__name = os.path.join(".", "MOBILE", "ios", "oonimkall.framework.zip")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isfile(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
|
|
|
engine.require("zip", "rm")
|
|
|
|
ooframework = OONIMKAllFramework()
|
|
|
|
ooframework.build(engine, options)
|
|
|
|
log("./make: building {}...".format(self.__name))
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"rm",
|
|
|
|
"-rf",
|
|
|
|
"oonimkall.framework.zip",
|
|
|
|
],
|
|
|
|
cwd=os.path.join(".", "MOBILE", "ios"),
|
|
|
|
)
|
|
|
|
engine.run(
|
|
|
|
[
|
|
|
|
"zip",
|
|
|
|
"-yr",
|
|
|
|
"oonimkall.framework.zip",
|
|
|
|
"oonimkall.framework",
|
|
|
|
],
|
|
|
|
cwd=os.path.join(".", "MOBILE", "ios"),
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class OONIMKAllPodspec:
|
|
|
|
"""OONIMKAllPodspec creates ./MOBILE/ios/oonimkall.podspec."""
|
|
|
|
|
|
|
|
__name = os.path.join(".", "MOBILE", "ios", "oonimkall.podspec")
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return self.__name
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
if os.path.isfile(self.__name) and not options.dry_run():
|
|
|
|
log("./make: {}: already built".format(self.__name))
|
|
|
|
return
|
|
|
|
version = datetime.datetime.now().strftime("%Y.%m.%d-%H%M%S")
|
|
|
|
engine.cat_sed_redirect(
|
|
|
|
"@VERSION@",
|
|
|
|
version,
|
|
|
|
os.path.join(".", "MOBILE", "template.podspec"),
|
|
|
|
self.__name,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class iOS:
|
|
|
|
"""iOS is the toplevel ios target."""
|
|
|
|
|
|
|
|
def name(self) -> str:
|
|
|
|
return "ios"
|
|
|
|
|
|
|
|
def build(self, engine: Engine, options: Options) -> None:
|
|
|
|
ooframeworkzip = OONIMKAllFrameworkZip()
|
|
|
|
ooframeworkzip.build(engine, options)
|
|
|
|
oopodspec = OONIMKAllPodspec()
|
|
|
|
oopodspec.build(engine, options)
|
|
|
|
|
|
|
|
|
2021-04-27 16:36:57 +02:00
|
|
|
TARGETS: List[Target] = [
|
|
|
|
Android(),
|
2021-04-29 10:30:39 +02:00
|
|
|
iOS(),
|
2021-04-27 16:36:57 +02:00
|
|
|
BundleJAR(),
|
|
|
|
OONIMKAllAAR(),
|
|
|
|
OONIProbePrivate(),
|
|
|
|
SDKAndroid(),
|
|
|
|
SDKGolangGo(),
|
|
|
|
SDKOONIGo(),
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
def main() -> None:
|
|
|
|
"""main function"""
|
|
|
|
alltargets: Dict[str, Target] = dict((t.name(), t) for t in TARGETS)
|
|
|
|
options = ConfigFromCLI.parse(list(alltargets.keys()))
|
|
|
|
engine = new_engine(options)
|
|
|
|
# note that we check whether the target is known in parse()
|
|
|
|
selected = alltargets[options.target()]
|
|
|
|
selected.build(engine, options)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|