forked from facebook/sapling
-
Notifications
You must be signed in to change notification settings - Fork 0
/
make-client.py
executable file
·165 lines (142 loc) · 5.54 KB
/
make-client.py
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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
#!/usr/bin/env python3
# Copyright (c) Meta Platforms, Inc. and affiliates.
#
# This software may be used and distributed according to the terms of the
# GNU General Public License version 2.
# This script generates the `eden` CLI executable.
# We use zipapp to bundle this as a single executable file.
# This script looks a bit complicated because the layout of
# the python modules in the source tree doesn't match their
# runtime names; we therefore massage them into the installation
# image, and pull in a couple of third party dependencies from pypi.
import argparse
import os
import shutil
import subprocess
import sys
import tempfile
import zipapp
from pipes import quote as shellquote
# Where to find the eden OSS directory; it contains this script.
DEFAULT_OSS_DIR = os.path.abspath(os.path.dirname(__file__))
# third party deps to include in the executable
DEPS = ["future", "six", "toml"]
# Source path to destination python module name.
# The lhs of each tuple is the path in the eden tree where the
# python sources are found, and the rhs is the destination path
MODULES = [
# Eden python libraries
("eden/fs/py/eden", "eden"),
# The cli
("eden/fs/cli", "eden/fs/cli"),
]
def run_cmd(cmd, env=None, cwd=None):
cmd_str = " ".join(shellquote(arg) for arg in cmd)
env_extra = env or {}
env = os.environ.copy()
print(
"+ "
+ " ".join(["%s=%s" % (k, shellquote(v)) for k, v in env_extra.items()])
+ " "
+ cmd_str
)
assert os.path.isfile(cmd[0]), cmd[0]
env.update(env_extra)
subprocess.check_call(cmd, env=env, cwd=cwd)
def generate_thrift_code(thrift_compiler, oss_dir, fb303_dir, gen_dir):
"""Generate python thrift clients for a couple of things"""
fb303_include_dir = os.path.join(fb303_dir, "include", "thrift-files")
thrift_files = [
os.path.join(oss_dir, "eden/fs/config/eden_config.thrift"),
os.path.join(oss_dir, "eden/fs/service/eden.thrift"),
os.path.join(fb303_include_dir, "fb303/thrift/fb303_core.thrift"),
os.path.join(oss_dir, "eden/fs/inodes/overlay/overlay.thrift"),
]
for t in thrift_files:
run_cmd(
[
thrift_compiler,
"-I",
oss_dir,
"-I",
fb303_include_dir,
"-gen",
"py:new_style",
"-out",
gen_dir,
t,
]
)
def copy_py(src_dir, instdir, dest_prefix):
"""Workhorse for processing the mapping from source tree to
installation image. This function copies only python files
from the source and places them under an alternative directory
structure in the destination"""
for root, _dirs, files in os.walk(src_dir):
rel_root = os.path.relpath(root, src_dir)
for f in files:
if f.endswith(".py"):
dest_dir = os.path.join(instdir, dest_prefix)
if rel_root != ".":
dest_dir = os.path.join(dest_dir, rel_root)
src_file_name = os.path.join(root, f)
dest_file_name = os.path.join(dest_dir, os.path.basename(f))
os.makedirs(dest_dir, exist_ok=True)
shutil.copyfile(src_file_name, dest_file_name)
def find_site_packages(instdir):
"""locate any and all site-packages directories in the install image"""
sp = []
for root, dirs, _files in os.walk(instdir):
for d in dirs:
if d == "site-packages":
sp.append(os.path.join(root, d))
return sp
def move_site_packages_to_root(instdir):
"""To reduce pythonpath headaches, after install packages from pip we
sweep them out of site-packages dirs and move them up to the root so
that they are reachable by the entrypoint in the zipapp"""
for sp in find_site_packages(instdir):
for child in os.listdir(sp):
os.rename(os.path.join(sp, child), os.path.join(instdir, child))
parser = argparse.ArgumentParser()
# Allow the caller to specify a specific python interpreter to use for the output
# application. This may help minimize headaches if the system python is upgraded and
# we want to use an alternate one.
parser.add_argument(
"--python", default=sys.executable, help="The python interpreter to use"
)
parser.add_argument(
"-o",
"--output",
default="eden.zip",
help="The output file location (default=%(default)s)",
)
parser.add_argument("--oss-dir", default=DEFAULT_OSS_DIR)
parser.add_argument("--fb303-dir")
parser.add_argument(
"--thrift-compiler",
default=os.path.join(DEFAULT_OSS_DIR, "external/install/bin/thrift1"),
)
parser.add_argument(
"--thrift-py",
default=os.path.join(DEFAULT_OSS_DIR, "external/fbthrift/thrift/lib/py"),
)
args = parser.parse_args()
with tempfile.TemporaryDirectory() as instdir:
generate_thrift_code(args.thrift_compiler, args.oss_dir, args.fb303_dir, instdir)
for src_dir, dest_prefix in MODULES:
copy_py(os.path.join(args.oss_dir, src_dir), instdir, dest_prefix)
# And the python thrift runtime
copy_py(args.thrift_py, instdir, "thrift")
for dep in DEPS:
# There's no supported way to call `pip` in process, so we just
# have to shell out and install it where we want it.
run_cmd([args.python, "-m", "pip", "install", dep, "--prefix", instdir])
move_site_packages_to_root(instdir)
# Generate the `eden` executable zipfile.
zipapp.create_archive(
instdir,
target=args.output,
interpreter=args.python,
main="eden.fs.cli.main:zipapp_main",
)