OLD | NEW |
| (Empty) |
1 # Copyright 2014 The Chromium Authors. All rights reserved. | |
2 # Use of this source code is governed by a BSD-style license that can be | |
3 # found in the LICENSE file. | |
4 | |
5 import fnmatch | |
6 import glob | |
7 import os | |
8 import shutil | |
9 import sys | |
10 import tempfile | |
11 | |
12 from devil.utils import cmd_helper | |
13 from pylib import constants | |
14 from pylib.constants import host_paths | |
15 | |
16 | |
17 _ISOLATE_SCRIPT = os.path.join( | |
18 host_paths.DIR_SOURCE_ROOT, 'tools', 'swarming_client', 'isolate.py') | |
19 | |
20 | |
21 def DefaultPathVariables(): | |
22 return { | |
23 'DEPTH': host_paths.DIR_SOURCE_ROOT, | |
24 'PRODUCT_DIR': constants.GetOutDirectory(), | |
25 } | |
26 | |
27 | |
28 def DefaultConfigVariables(): | |
29 # Note: This list must match the --config-vars in build/isolate.gypi | |
30 return { | |
31 'CONFIGURATION_NAME': constants.GetBuildType(), | |
32 'OS': 'android', | |
33 'asan': '0', | |
34 'branding': 'Chromium', | |
35 'chromeos': '0', | |
36 'component': 'static_library', | |
37 'enable_pepper_cdms': '0', | |
38 'enable_plugins': '0', | |
39 'fastbuild': '0', | |
40 'icu_use_data_file_flag': '1', | |
41 'kasko': '0', | |
42 'lsan': '0', | |
43 'msan': '0', | |
44 # TODO(maruel): This may not always be true. | |
45 'target_arch': 'arm', | |
46 'tsan': '0', | |
47 'use_custom_libcxx': '0', | |
48 'use_instrumented_libraries': '0', | |
49 'use_prebuilt_instrumented_libraries': '0', | |
50 'use_ozone': '0', | |
51 'use_x11': '0', | |
52 'v8_use_external_startup_data': '1', | |
53 'msvs_version': '0', | |
54 } | |
55 | |
56 | |
57 def IsIsolateEmpty(isolate_path): | |
58 """Returns whether there are no files in the .isolate.""" | |
59 with open(isolate_path) as f: | |
60 return "'files': []" in f.read() | |
61 | |
62 | |
63 class Isolator(object): | |
64 """Manages calls to isolate.py for the android test runner scripts.""" | |
65 | |
66 def __init__(self): | |
67 self._isolate_deps_dir = tempfile.mkdtemp() | |
68 | |
69 @property | |
70 def isolate_deps_dir(self): | |
71 return self._isolate_deps_dir | |
72 | |
73 def Clear(self): | |
74 """Deletes the isolate dependency directory.""" | |
75 if os.path.exists(self._isolate_deps_dir): | |
76 shutil.rmtree(self._isolate_deps_dir) | |
77 | |
78 def Remap(self, isolate_abs_path, isolated_abs_path, | |
79 path_variables=None, config_variables=None): | |
80 """Remaps data dependencies into |self._isolate_deps_dir|. | |
81 | |
82 Args: | |
83 isolate_abs_path: The absolute path to the .isolate file, which specifies | |
84 data dependencies in the source tree. | |
85 isolated_abs_path: The absolute path to the .isolated file, which is | |
86 generated by isolate.py and specifies data dependencies in | |
87 |self._isolate_deps_dir| and their digests. | |
88 path_variables: A dict containing everything that should be passed | |
89 as a |--path-variable| to the isolate script. Defaults to the return | |
90 value of |DefaultPathVariables()|. | |
91 config_variables: A dict containing everything that should be passed | |
92 as a |--config-variable| to the isolate script. Defaults to the return | |
93 value of |DefaultConfigVariables()|. | |
94 Raises: | |
95 Exception if the isolate command fails for some reason. | |
96 """ | |
97 if not path_variables: | |
98 path_variables = DefaultPathVariables() | |
99 if not config_variables: | |
100 config_variables = DefaultConfigVariables() | |
101 | |
102 isolate_cmd = [ | |
103 sys.executable, _ISOLATE_SCRIPT, 'remap', | |
104 '--isolate', isolate_abs_path, | |
105 '--isolated', isolated_abs_path, | |
106 '--outdir', self._isolate_deps_dir, | |
107 ] | |
108 for k, v in path_variables.iteritems(): | |
109 isolate_cmd.extend(['--path-variable', k, v]) | |
110 for k, v in config_variables.iteritems(): | |
111 isolate_cmd.extend(['--config-variable', k, v]) | |
112 | |
113 exit_code, _ = cmd_helper.GetCmdStatusAndOutput(isolate_cmd) | |
114 if exit_code: | |
115 raise Exception('isolate command failed: %s' % ' '.join(isolate_cmd)) | |
116 | |
117 def VerifyHardlinks(self): | |
118 """Checks |isolate_deps_dir| for a hardlink. | |
119 | |
120 Returns: | |
121 True if a hardlink is found. | |
122 False if nothing is found. | |
123 Raises: | |
124 Exception if a non-hardlink is found. | |
125 """ | |
126 for root, _, filenames in os.walk(self._isolate_deps_dir): | |
127 if filenames: | |
128 linked_file = os.path.join(root, filenames[0]) | |
129 orig_file = os.path.join( | |
130 self._isolate_deps_dir, | |
131 os.path.relpath(linked_file, self._isolate_deps_dir)) | |
132 if os.stat(linked_file).st_ino == os.stat(orig_file).st_ino: | |
133 return True | |
134 else: | |
135 raise Exception('isolate remap command did not use hardlinks.') | |
136 return False | |
137 | |
138 def PurgeExcluded(self, deps_exclusion_list): | |
139 """Deletes anything on |deps_exclusion_list| from |self._isolate_deps_dir|. | |
140 | |
141 Args: | |
142 deps_exclusion_list: A list of globs to exclude from the isolate | |
143 dependency directory. | |
144 """ | |
145 excluded_paths = ( | |
146 x for y in deps_exclusion_list | |
147 for x in glob.glob( | |
148 os.path.abspath(os.path.join(self._isolate_deps_dir, y)))) | |
149 for p in excluded_paths: | |
150 if os.path.isdir(p): | |
151 shutil.rmtree(p) | |
152 else: | |
153 os.remove(p) | |
154 | |
155 @classmethod | |
156 def _DestructiveMerge(cls, src, dest): | |
157 if os.path.exists(dest) and os.path.isdir(dest): | |
158 for p in os.listdir(src): | |
159 cls._DestructiveMerge(os.path.join(src, p), os.path.join(dest, p)) | |
160 os.rmdir(src) | |
161 else: | |
162 shutil.move(src, dest) | |
163 | |
164 | |
165 def MoveOutputDeps(self): | |
166 """Moves files from the output directory to the top level of | |
167 |self._isolate_deps_dir|. | |
168 | |
169 Moves pak files from the output directory to to <isolate_deps_dir>/paks | |
170 Moves files from the product directory to <isolate_deps_dir> | |
171 """ | |
172 # On Android, all pak files need to be in the top-level 'paks' directory. | |
173 paks_dir = os.path.join(self._isolate_deps_dir, 'paks') | |
174 os.mkdir(paks_dir) | |
175 | |
176 deps_out_dir = os.path.join( | |
177 self._isolate_deps_dir, | |
178 os.path.relpath(os.path.join(constants.GetOutDirectory(), os.pardir), | |
179 host_paths.DIR_SOURCE_ROOT)) | |
180 for root, _, filenames in os.walk(deps_out_dir): | |
181 for filename in fnmatch.filter(filenames, '*.pak'): | |
182 shutil.move(os.path.join(root, filename), paks_dir) | |
183 | |
184 # Move everything in PRODUCT_DIR to top level. | |
185 deps_product_dir = os.path.join( | |
186 deps_out_dir, os.path.basename(constants.GetOutDirectory())) | |
187 if os.path.isdir(deps_product_dir): | |
188 for p in os.listdir(deps_product_dir): | |
189 Isolator._DestructiveMerge(os.path.join(deps_product_dir, p), | |
190 os.path.join(self._isolate_deps_dir, p)) | |
191 os.rmdir(deps_product_dir) | |
192 os.rmdir(deps_out_dir) | |
OLD | NEW |