blob: 43153c60644269704c5c7ffd2d2a9f7672fbcfc4 [file] [log] [blame]
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +01001#!/usr/bin/env python
2# Copyright (c) 2018, the R8 project authors. Please see the AUTHORS file
3# for details. All rights reserved. Use of this source code is governed by a
4# BSD-style license that can be found in the LICENSE file.
5
Christoffer Quist Adamsen5c9ded12021-01-14 14:29:37 +01006import utils
7
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +01008from distutils.version import LooseVersion
9import os
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +010010import shutil
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010011
Christoffer Quist Adamsen5c9ded12021-01-14 14:29:37 +010012if utils.is_python3():
13 from html.parser import HTMLParser
14else:
15 from HTMLParser import HTMLParser
16
17
Morten Krogh-Jespersend45d95e2019-02-07 13:27:17 +010018def add_r8_dependency(checkout_dir, temp_dir, minified):
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010019 build_file = os.path.join(checkout_dir, 'build.gradle')
Christoffer Quist Adamsenbbe5d7d2019-01-23 13:15:21 +010020 assert os.path.isfile(build_file), (
21 'Expected a file to be present at {}'.format(build_file))
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010022
23 with open(build_file) as f:
24 lines = f.readlines()
25
26 added_r8_dependency = False
27 is_inside_dependencies = False
28
29 with open(build_file, 'w') as f:
30 gradle_version = None
31 for line in lines:
32 stripped = line.strip()
33 if stripped == 'dependencies {':
Christoffer Quist Adamsenbbe5d7d2019-01-23 13:15:21 +010034 assert not is_inside_dependencies, (
35 'Unexpected line with \'dependencies {\'')
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010036 is_inside_dependencies = True
37 if is_inside_dependencies:
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +010038 if '/r8.jar' in stripped or '/r8lib.jar' in stripped:
39 # Skip line to avoid dependency on r8.jar
40 continue
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010041 elif 'com.android.tools.build:gradle:' in stripped:
42 gradle_version = stripped[stripped.rindex(':')+1:-1]
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +010043 indent = ''.ljust(line.index('classpath'))
Morten Krogh-Jespersend45d95e2019-02-07 13:27:17 +010044 jar = os.path.join(temp_dir, 'r8lib.jar' if minified else 'r8.jar')
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +010045 f.write('{}classpath files(\'{}\')\n'.format(indent, jar))
46 added_r8_dependency = True
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010047 elif stripped == '}':
48 is_inside_dependencies = False
49 f.write(line)
50
51 assert added_r8_dependency, 'Unable to add R8 as a dependency'
52 assert gradle_version
53 assert LooseVersion(gradle_version) >= LooseVersion('3.2'), (
54 'Unsupported gradle version: {} (must use at least gradle '
55 + 'version 3.2)').format(gradle_version)
56
Morten Krogh-Jespersende566ea2019-02-18 11:59:48 +010057def add_settings_gradle(checkout_dir, name):
58 settings_file = os.path.join(checkout_dir, 'settings.gradle')
59 if os.path.isfile(settings_file):
60 return
61
62 with open(settings_file, "w+") as f:
63 f.write("rootProject.name = '{}'\n".format(name))
64
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010065def remove_r8_dependency(checkout_dir):
66 build_file = os.path.join(checkout_dir, 'build.gradle')
Christoffer Quist Adamsen4038bbe2019-01-15 14:31:46 +010067 assert os.path.isfile(build_file), (
68 'Expected a file to be present at {}'.format(build_file))
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010069 with open(build_file) as f:
70 lines = f.readlines()
71 with open(build_file, 'w') as f:
72 for line in lines:
Christoffer Quist Adamsenbbe5d7d2019-01-23 13:15:21 +010073 if ('/r8.jar' not in line) and ('/r8lib.jar' not in line):
Christoffer Quist Adamsen10b7db82018-12-13 14:50:38 +010074 f.write(line)
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +010075
Christoffer Quist Adamsenb899c112019-03-06 15:25:00 +010076def GetMinAndCompileSdk(app, checkout_dir, apk_reference):
77 compile_sdk = app.compile_sdk
78 min_sdk = app.min_sdk
Christoffer Quist Adamsene59840b2019-01-22 15:25:15 +010079
Christoffer Quist Adamsenbbe5d7d2019-01-23 13:15:21 +010080 if not compile_sdk or not min_sdk:
Christoffer Quist Adamsenb899c112019-03-06 15:25:00 +010081 build_gradle_file = os.path.join(checkout_dir, app.module, 'build.gradle')
Christoffer Quist Adamsenbbe5d7d2019-01-23 13:15:21 +010082 assert os.path.isfile(build_gradle_file), (
83 'Expected to find build.gradle file at {}'.format(build_gradle_file))
Christoffer Quist Adamsene59840b2019-01-22 15:25:15 +010084
Christoffer Quist Adamsenbbe5d7d2019-01-23 13:15:21 +010085 # Attempt to find the sdk values from build.gradle.
86 with open(build_gradle_file) as f:
87 for line in f.readlines():
88 stripped = line.strip()
89 if stripped.startswith('compileSdkVersion '):
Christoffer Quist Adamsenb899c112019-03-06 15:25:00 +010090 if not app.compile_sdk:
Christoffer Quist Adamsenbbe5d7d2019-01-23 13:15:21 +010091 assert not compile_sdk
92 compile_sdk = int(stripped[len('compileSdkVersion '):])
93 elif stripped.startswith('minSdkVersion '):
Christoffer Quist Adamsenb899c112019-03-06 15:25:00 +010094 if not app.min_sdk:
Christoffer Quist Adamsenbbe5d7d2019-01-23 13:15:21 +010095 assert not min_sdk
96 min_sdk = int(stripped[len('minSdkVersion '):])
Christoffer Quist Adamsene59840b2019-01-22 15:25:15 +010097
98 assert min_sdk, (
99 'Expected to find `minSdkVersion` in {}'.format(build_gradle_file))
100 assert compile_sdk, (
101 'Expected to find `compileSdkVersion` in {}'.format(build_gradle_file))
102
Christoffer Quist Adamsene59840b2019-01-22 15:25:15 +0100103 return (min_sdk, compile_sdk)
104
Christoffer Quist Adamsen81321b62019-01-15 14:39:53 +0100105def IsGradleTaskName(x):
106 # Check that it is non-empty.
107 if not x:
108 return False
109 # Check that there is no whitespace.
110 for c in x:
111 if c.isspace():
112 return False
113 # Check that the first character following an optional ':' is a lower-case
114 # alphabetic character.
115 c = x[0]
116 if c == ':' and len(x) >= 2:
117 c = x[1]
118 return c.isalpha() and c.islower()
119
120def IsGradleCompilerTask(x, shrinker):
121 if 'r8' in shrinker:
122 assert 'transformClassesWithDexBuilderFor' not in x
123 assert 'transformDexArchiveWithDexMergerFor' not in x
124 return 'transformClassesAndResourcesWithR8For' in x
125
Morten Krogh-Jespersen19ffe182019-02-15 10:12:31 +0100126 assert shrinker == 'pg'
Christoffer Quist Adamsen81321b62019-01-15 14:39:53 +0100127 return ('transformClassesAndResourcesWithProguard' in x
128 or 'transformClassesWithDexBuilderFor' in x
129 or 'transformDexArchiveWithDexMergerFor' in x)
130
Christoffer Quist Adamsen7d74ef72019-04-12 10:33:38 +0200131def ListFiles(directory, predicate=None):
132 files = []
133 for root, directories, filenames in os.walk(directory):
134 for filename in filenames:
135 file = os.path.join(root, filename)
136 if predicate is None or predicate(file):
137 files.append(file)
138 return files
139
Christoffer Quist Adamsenb899c112019-03-06 15:25:00 +0100140def SetPrintConfigurationDirective(app, checkout_dir, destination):
141 proguard_config_file = FindProguardConfigurationFile(app, checkout_dir)
Christoffer Quist Adamsen4038bbe2019-01-15 14:31:46 +0100142 with open(proguard_config_file) as f:
143 lines = f.readlines()
144 with open(proguard_config_file, 'w') as f:
145 for line in lines:
146 if '-printconfiguration' not in line:
147 f.write(line)
Morten Krogh-Jespersene1aeead2019-01-29 11:16:44 +0100148 # Check that there is a line-break at the end of the file or insert one.
Christoffer Quist Adamsenc479a422019-02-07 10:23:50 +0100149 if len(lines) and lines[-1].strip():
Morten Krogh-Jespersene1aeead2019-01-29 11:16:44 +0100150 f.write('\n')
Christoffer Quist Adamsen4038bbe2019-01-15 14:31:46 +0100151 f.write('-printconfiguration {}\n'.format(destination))
152
Christoffer Quist Adamsenb899c112019-03-06 15:25:00 +0100153def FindProguardConfigurationFile(app, checkout_dir):
Christoffer Quist Adamsen493acd82019-03-14 12:20:02 +0100154 candidates = [
155 'proguard.cfg',
156 'proguard-rules.pro',
157 'proguard-rules.txt',
158 'proguard-project.txt']
Christoffer Quist Adamsen4038bbe2019-01-15 14:31:46 +0100159 for candidate in candidates:
Christoffer Quist Adamsenb899c112019-03-06 15:25:00 +0100160 proguard_config_file = os.path.join(checkout_dir, app.module, candidate)
Christoffer Quist Adamsen4038bbe2019-01-15 14:31:46 +0100161 if os.path.isfile(proguard_config_file):
162 return proguard_config_file
163 # Currently assuming that the Proguard configuration file can be found at
164 # one of the predefined locations.
Christoffer Quist Adamsen493acd82019-03-14 12:20:02 +0100165 assert False, 'Unable to find Proguard configuration file'
Christoffer Quist Adamsen4038bbe2019-01-15 14:31:46 +0100166
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +0100167def Move(src, dst, quiet=False):
168 if not quiet:
169 print('Moving `{}` to `{}`'.format(src, dst))
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100170 dst_parent = os.path.dirname(dst)
171 if not os.path.isdir(dst_parent):
172 os.makedirs(dst_parent)
173 elif os.path.isdir(dst):
174 shutil.rmtree(dst)
175 elif os.path.isfile(dst):
176 os.remove(dst)
177 os.rename(src, dst)
178
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +0100179def MoveDir(src, dst, quiet=False):
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100180 assert os.path.isdir(src)
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +0100181 Move(src, dst, quiet=quiet)
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100182
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +0100183def MoveFile(src, dst, quiet=False):
Christoffer Quist Adamsenc479a422019-02-07 10:23:50 +0100184 assert os.path.isfile(src), "Expected a file to be present at " + src
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +0100185 Move(src, dst, quiet=quiet)
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100186
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +0100187def MoveProfileReportTo(dest_dir, build_stdout, quiet=False):
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100188 html_file = None
189 profile_message = 'See the profiling report at: '
Morten Krogh-Jespersenb5d45cd2019-03-19 13:58:30 +0100190 # We are not interested in the profiling report for buildSrc.
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100191 for line in build_stdout:
Morten Krogh-Jespersenb5d45cd2019-03-19 13:58:30 +0100192 if (profile_message in line) and ('buildSrc' not in line):
193 assert not html_file, "Only one report should be created"
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100194 html_file = line[len(profile_message):]
195 if html_file.startswith('file://'):
196 html_file = html_file[len('file://'):]
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100197
198 if not html_file:
199 return
200
201 assert os.path.isfile(html_file), 'Expected to find HTML file at {}'.format(
202 html_file)
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +0100203 MoveFile(html_file, os.path.join(dest_dir, 'index.html'), quiet=quiet)
Christoffer Quist Adamsen1d87ca62019-01-11 12:22:26 +0100204
205 html_dir = os.path.dirname(html_file)
206 for dir_name in ['css', 'js']:
Christoffer Quist Adamsen1de3dde2019-01-24 13:17:46 +0100207 MoveDir(os.path.join(html_dir, dir_name), os.path.join(dest_dir, dir_name),
208 quiet=quiet)
Christoffer Quist Adamsen81321b62019-01-15 14:39:53 +0100209
Christoffer Quist Adamsen7cf4c562019-03-07 10:57:33 +0100210def MoveXMLTestResultFileTo(xml_test_result_dest, test_stdout, quiet=False):
211 xml_test_result_file = None
212 xml_result_reporter_message = 'XML test result file generated at '
213 for line in test_stdout:
214 if xml_result_reporter_message in line:
215 index_from = (
216 line.index(xml_result_reporter_message)
217 + len(xml_result_reporter_message))
218 index_to = line.index('.xml') + len('.xml')
219 xml_test_result_file = line[index_from:index_to]
220 break
221
222 assert os.path.isfile(xml_test_result_file), (
223 'Expected to find XML file at {}'.format(xml_test_result_file))
224
225 MoveFile(xml_test_result_file, xml_test_result_dest, quiet=quiet)
226
Christoffer Quist Adamsen81321b62019-01-15 14:39:53 +0100227def ParseProfileReport(profile_dir):
228 html_file = os.path.join(profile_dir, 'index.html')
229 assert os.path.isfile(html_file)
230
231 parser = ProfileReportParser()
232 with open(html_file) as f:
233 for line in f.readlines():
234 parser.feed(line)
235 return parser.result
236
237# A simple HTML parser that recognizes the following pattern:
238#
239# <tr>
240# <td class="indentPath">:app:transformClassesAndResourcesWithR8ForRelease</td>
241# <td class="numeric">3.490s</td>
242# <td></td>
243# </tr>
244class ProfileReportParser(HTMLParser):
Christoffer Quist Adamsend0e0a7c2019-01-22 10:00:30 +0100245 def __init__(self):
246 HTMLParser.__init__(self)
247 self.entered_table_row = False
248 self.entered_task_name_cell = False
249 self.entered_duration_cell = False
Christoffer Quist Adamsen81321b62019-01-15 14:39:53 +0100250
Christoffer Quist Adamsend0e0a7c2019-01-22 10:00:30 +0100251 self.current_task_name = None
252 self.current_duration = None
Christoffer Quist Adamsen81321b62019-01-15 14:39:53 +0100253
Christoffer Quist Adamsend0e0a7c2019-01-22 10:00:30 +0100254 self.result = {}
Christoffer Quist Adamsen81321b62019-01-15 14:39:53 +0100255
256 def handle_starttag(self, tag, attrs):
257 entered_table_row_before = self.entered_table_row
258 entered_task_name_cell_before = self.entered_task_name_cell
259
260 self.entered_table_row = (tag == 'tr')
261 self.entered_task_name_cell = (tag == 'td' and entered_table_row_before)
262 self.entered_duration_cell = (
263 self.current_task_name
264 and tag == 'td'
265 and entered_task_name_cell_before)
266
267 def handle_endtag(self, tag):
268 if tag == 'tr':
269 if self.current_task_name and self.current_duration:
270 self.result[self.current_task_name] = self.current_duration
271 self.current_task_name = None
272 self.current_duration = None
273 self.entered_table_row = False
274
275 def handle_data(self, data):
276 stripped = data.strip()
277 if not stripped:
278 return
279 if self.entered_task_name_cell:
280 if IsGradleTaskName(stripped):
281 self.current_task_name = stripped
282 elif self.entered_duration_cell and stripped.endswith('s'):
Christoffer Quist Adamsend0e0a7c2019-01-22 10:00:30 +0100283 duration = stripped[:-1]
284 if 'm' in duration:
285 tmp = duration.split('m')
286 minutes = int(tmp[0])
287 seconds = float(tmp[1])
288 else:
289 minutes = 0
290 seconds = float(duration)
291 self.current_duration = 60 * minutes + seconds
Christoffer Quist Adamsen81321b62019-01-15 14:39:53 +0100292 self.entered_table_row = False