diff options
-rwxr-xr-x | scripts/contrib/oe-build-perf-report-email.py | 266 | ||||
-rw-r--r-- | scripts/lib/build_perf/scrape-html-report.js | 56 |
2 files changed, 322 insertions, 0 deletions
diff --git a/scripts/contrib/oe-build-perf-report-email.py b/scripts/contrib/oe-build-perf-report-email.py new file mode 100755 index 0000000000..7f4274efed --- /dev/null +++ b/scripts/contrib/oe-build-perf-report-email.py | |||
@@ -0,0 +1,266 @@ | |||
1 | #!/usr/bin/python3 | ||
2 | # | ||
3 | # Send build performance test report emails | ||
4 | # | ||
5 | # Copyright (c) 2017, Intel Corporation. | ||
6 | # | ||
7 | # This program is free software; you can redistribute it and/or modify it | ||
8 | # under the terms and conditions of the GNU General Public License, | ||
9 | # version 2, as published by the Free Software Foundation. | ||
10 | # | ||
11 | # This program is distributed in the hope it will be useful, but WITHOUT | ||
12 | # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or | ||
13 | # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for | ||
14 | # more details. | ||
15 | # | ||
16 | import argparse | ||
17 | import base64 | ||
18 | import logging | ||
19 | import os | ||
20 | import pwd | ||
21 | import re | ||
22 | import shutil | ||
23 | import smtplib | ||
24 | import subprocess | ||
25 | import sys | ||
26 | import tempfile | ||
27 | from email.mime.multipart import MIMEMultipart | ||
28 | from email.mime.text import MIMEText | ||
29 | |||
30 | |||
31 | # Setup logging | ||
32 | logging.basicConfig(level=logging.INFO, format="%(levelname)s: %(message)s") | ||
33 | log = logging.getLogger('oe-build-perf-report') | ||
34 | |||
35 | |||
36 | # Find js scaper script | ||
37 | SCRAPE_JS = os.path.join(os.path.dirname(__file__), '..', 'lib', 'build_perf', | ||
38 | 'scrape-html-report.js') | ||
39 | if not os.path.isfile(SCRAPE_JS): | ||
40 | log.error("Unableto find oe-build-perf-report-scrape.js") | ||
41 | sys.exit(1) | ||
42 | |||
43 | |||
44 | class ReportError(Exception): | ||
45 | """Local errors""" | ||
46 | pass | ||
47 | |||
48 | |||
49 | def check_utils(): | ||
50 | """Check that all needed utils are installed in the system""" | ||
51 | missing = [] | ||
52 | for cmd in ('phantomjs', 'optipng'): | ||
53 | if not shutil.which(cmd): | ||
54 | missing.append(cmd) | ||
55 | if missing: | ||
56 | log.error("The following tools are missing: %s", ' '.join(missing)) | ||
57 | sys.exit(1) | ||
58 | |||
59 | |||
60 | def parse_args(argv): | ||
61 | """Parse command line arguments""" | ||
62 | description = """Email build perf test report""" | ||
63 | parser = argparse.ArgumentParser( | ||
64 | formatter_class=argparse.ArgumentDefaultsHelpFormatter, | ||
65 | description=description) | ||
66 | |||
67 | parser.add_argument('--debug', '-d', action='store_true', | ||
68 | help="Verbose logging") | ||
69 | parser.add_argument('--quiet', '-q', action='store_true', | ||
70 | help="Only print errors") | ||
71 | parser.add_argument('--to', action='append', | ||
72 | help="Recipients of the email") | ||
73 | parser.add_argument('--subject', default="Yocto build perf test report", | ||
74 | help="Email subject") | ||
75 | parser.add_argument('--outdir', '-o', | ||
76 | help="Store files in OUTDIR. Can be used to preserve " | ||
77 | "the email parts") | ||
78 | parser.add_argument('--text', | ||
79 | help="Plain text message") | ||
80 | parser.add_argument('--html', | ||
81 | help="HTML peport generated by oe-build-perf-report") | ||
82 | parser.add_argument('--phantomjs-args', action='append', | ||
83 | help="Extra command line arguments passed to PhantomJS") | ||
84 | |||
85 | args = parser.parse_args(argv) | ||
86 | |||
87 | if not args.html and not args.text: | ||
88 | parser.error("Please specify --html and/or --text") | ||
89 | |||
90 | return args | ||
91 | |||
92 | |||
93 | def decode_png(infile, outfile): | ||
94 | """Parse/decode/optimize png data from a html element""" | ||
95 | with open(infile) as f: | ||
96 | raw_data = f.read() | ||
97 | |||
98 | # Grab raw base64 data | ||
99 | b64_data = re.sub('^.*href="data:image/png;base64,', '', raw_data, 1) | ||
100 | b64_data = re.sub('">.+$', '', b64_data, 1) | ||
101 | |||
102 | # Replace file with proper decoded png | ||
103 | with open(outfile, 'wb') as f: | ||
104 | f.write(base64.b64decode(b64_data)) | ||
105 | |||
106 | subprocess.check_output(['optipng', outfile], stderr=subprocess.STDOUT) | ||
107 | |||
108 | |||
109 | def encode_png(pngfile): | ||
110 | """Encode png into a <img> html element""" | ||
111 | with open(pngfile, 'rb') as f: | ||
112 | data = f.read() | ||
113 | |||
114 | b64_data = base64.b64encode(data) | ||
115 | return '<img src="data:image/png;base64,' + b64_data.decode('utf-8') + '">\n' | ||
116 | |||
117 | |||
118 | def mangle_html_report(infile, outfile, pngs): | ||
119 | """Mangle html file into a email compatible format""" | ||
120 | paste = True | ||
121 | png_dir = os.path.dirname(outfile) | ||
122 | with open(infile) as f_in: | ||
123 | with open(outfile, 'w') as f_out: | ||
124 | for line in f_in.readlines(): | ||
125 | stripped = line.strip() | ||
126 | # Strip out scripts | ||
127 | if stripped == '<!--START-OF-SCRIPTS-->': | ||
128 | paste = False | ||
129 | elif stripped == '<!--END-OF-SCRIPTS-->': | ||
130 | paste = True | ||
131 | elif paste: | ||
132 | if re.match('^.+href="data:image/png;base64', stripped): | ||
133 | # Strip out encoded pngs (as they're huge in size) | ||
134 | continue | ||
135 | elif 'www.gstatic.com' in stripped: | ||
136 | # HACK: drop references to external static pages | ||
137 | continue | ||
138 | |||
139 | # Replace charts with <img> elements | ||
140 | match = re.match('<div id="(?P<id>\w+)"', stripped) | ||
141 | if match and match.group('id') in pngs: | ||
142 | #f_out.write('<img src="{}">\n'.format(match.group('id') + '.png')) | ||
143 | png_file = os.path.join(png_dir, match.group('id') + '.png') | ||
144 | f_out.write(encode_png(png_file)) | ||
145 | else: | ||
146 | f_out.write(line) | ||
147 | |||
148 | |||
149 | def scrape_html_report(report, outdir, phantomjs_extra_args=None): | ||
150 | """Scrape html report into a format sendable by email""" | ||
151 | tmpdir = tempfile.mkdtemp(dir='.') | ||
152 | log.debug("Using tmpdir %s for phantomjs output", tmpdir) | ||
153 | |||
154 | if not os.path.isdir(outdir): | ||
155 | os.mkdir(outdir) | ||
156 | if os.path.splitext(report)[1] not in ('.html', '.htm'): | ||
157 | raise ReportError("Invalid file extension for report, needs to be " | ||
158 | "'.html' or '.htm'") | ||
159 | |||
160 | try: | ||
161 | log.info("Scraping HTML report with PhangomJS") | ||
162 | extra_args = phantomjs_extra_args if phantomjs_extra_args else [] | ||
163 | subprocess.check_output(['phantomjs', '--debug=true'] + extra_args + | ||
164 | [SCRAPE_JS, report, tmpdir], | ||
165 | stderr=subprocess.STDOUT) | ||
166 | |||
167 | pngs = [] | ||
168 | attachments = [] | ||
169 | for fname in os.listdir(tmpdir): | ||
170 | base, ext = os.path.splitext(fname) | ||
171 | if ext == '.png': | ||
172 | log.debug("Decoding %s", fname) | ||
173 | decode_png(os.path.join(tmpdir, fname), | ||
174 | os.path.join(outdir, fname)) | ||
175 | pngs.append(base) | ||
176 | attachments.append(fname) | ||
177 | elif ext in ('.html', '.htm'): | ||
178 | report_file = fname | ||
179 | else: | ||
180 | log.warning("Unknown file extension: '%s'", ext) | ||
181 | #shutil.move(os.path.join(tmpdir, fname), outdir) | ||
182 | |||
183 | log.debug("Mangling html report file %s", report_file) | ||
184 | mangle_html_report(os.path.join(tmpdir, report_file), | ||
185 | os.path.join(outdir, report_file), pngs) | ||
186 | return report_file, attachments | ||
187 | finally: | ||
188 | shutil.rmtree(tmpdir) | ||
189 | |||
190 | def send_email(text_fn, html_fn, subject, recipients): | ||
191 | """Send email""" | ||
192 | # Generate email message | ||
193 | text_msg = html_msg = None | ||
194 | if text_fn: | ||
195 | with open(text_fn) as f: | ||
196 | text_msg = MIMEText("Yocto build performance test report.\n" + | ||
197 | f.read(), 'plain') | ||
198 | if html_fn: | ||
199 | with open(html_fn) as f: | ||
200 | html_msg = MIMEText(f.read(), 'html') | ||
201 | |||
202 | if text_msg and html_msg: | ||
203 | msg = MIMEMultipart('alternative') | ||
204 | msg.attach(text_msg) | ||
205 | msg.attach(html_msg) | ||
206 | elif text_msg: | ||
207 | msg = text_msg | ||
208 | elif html_msg: | ||
209 | msg = html_msg | ||
210 | else: | ||
211 | raise ReportError("Neither plain text nor html body specified") | ||
212 | |||
213 | full_name = pwd.getpwuid(os.getuid()).pw_gecos.split(',')[0] | ||
214 | email = os.environ.get('EMAIL', os.getlogin()) | ||
215 | msg['From'] = "{} <{}>".format(full_name, email) | ||
216 | msg['To'] = ', '.join(recipients) | ||
217 | msg['Subject'] = subject | ||
218 | |||
219 | # Send email | ||
220 | with smtplib.SMTP('localhost') as smtp: | ||
221 | smtp.send_message(msg) | ||
222 | |||
223 | |||
224 | def main(argv=None): | ||
225 | """Script entry point""" | ||
226 | args = parse_args(argv) | ||
227 | if args.quiet: | ||
228 | log.setLevel(logging.ERROR) | ||
229 | if args.debug: | ||
230 | log.setLevel(logging.DEBUG) | ||
231 | |||
232 | check_utils() | ||
233 | |||
234 | if args.outdir: | ||
235 | outdir = args.outdir | ||
236 | if not os.path.exists(outdir): | ||
237 | os.mkdir(outdir) | ||
238 | else: | ||
239 | outdir = tempfile.mkdtemp(dir='.') | ||
240 | |||
241 | try: | ||
242 | log.debug("Storing email parts in %s", outdir) | ||
243 | html_report = None | ||
244 | if args.html: | ||
245 | scrape_html_report(args.html, outdir, args.phantomjs_args) | ||
246 | html_report = os.path.join(outdir, args.html) | ||
247 | |||
248 | if args.to: | ||
249 | log.info("Sending email to %s", ', '.join(args.to)) | ||
250 | send_email(args.text, html_report, args.subject, args.to) | ||
251 | except subprocess.CalledProcessError as err: | ||
252 | log.error("%s, with output:\n%s", str(err), err.output.decode()) | ||
253 | return 1 | ||
254 | except ReportError as err: | ||
255 | log.error(err) | ||
256 | return 1 | ||
257 | finally: | ||
258 | if not args.outdir: | ||
259 | log.debug("Wiping %s", outdir) | ||
260 | shutil.rmtree(outdir) | ||
261 | |||
262 | return 0 | ||
263 | |||
264 | |||
265 | if __name__ == "__main__": | ||
266 | sys.exit(main()) | ||
diff --git a/scripts/lib/build_perf/scrape-html-report.js b/scripts/lib/build_perf/scrape-html-report.js new file mode 100644 index 0000000000..05a1f57001 --- /dev/null +++ b/scripts/lib/build_perf/scrape-html-report.js | |||
@@ -0,0 +1,56 @@ | |||
1 | var fs = require('fs'); | ||
2 | var system = require('system'); | ||
3 | var page = require('webpage').create(); | ||
4 | |||
5 | // Examine console log for message from chart drawing | ||
6 | page.onConsoleMessage = function(msg) { | ||
7 | console.log(msg); | ||
8 | if (msg === "ALL CHARTS READY") { | ||
9 | window.charts_ready = true; | ||
10 | } | ||
11 | else if (msg.slice(0, 11) === "CHART READY") { | ||
12 | var chart_id = msg.split(" ")[2]; | ||
13 | console.log('grabbing ' + chart_id); | ||
14 | var png_data = page.evaluate(function (chart_id) { | ||
15 | var chart_div = document.getElementById(chart_id + '_png'); | ||
16 | return chart_div.outerHTML; | ||
17 | }, chart_id); | ||
18 | fs.write(args[2] + '/' + chart_id + '.png', png_data, 'w'); | ||
19 | } | ||
20 | }; | ||
21 | |||
22 | // Check command line arguments | ||
23 | var args = system.args; | ||
24 | if (args.length != 3) { | ||
25 | console.log("USAGE: " + args[0] + " REPORT_HTML OUT_DIR\n"); | ||
26 | phantom.exit(1); | ||
27 | } | ||
28 | |||
29 | // Open the web page | ||
30 | page.open(args[1], function(status) { | ||
31 | if (status == 'fail') { | ||
32 | console.log("Failed to open file '" + args[1] + "'"); | ||
33 | phantom.exit(1); | ||
34 | } | ||
35 | }); | ||
36 | |||
37 | // Check status every 100 ms | ||
38 | interval = window.setInterval(function () { | ||
39 | //console.log('waiting'); | ||
40 | if (window.charts_ready) { | ||
41 | clearTimeout(timer); | ||
42 | clearInterval(interval); | ||
43 | |||
44 | var fname = args[1].replace(/\/+$/, "").split("/").pop() | ||
45 | console.log("saving " + fname); | ||
46 | fs.write(args[2] + '/' + fname, page.content, 'w'); | ||
47 | phantom.exit(0); | ||
48 | } | ||
49 | }, 100); | ||
50 | |||
51 | // Time-out after 10 seconds | ||
52 | timer = window.setTimeout(function () { | ||
53 | clearInterval(interval); | ||
54 | console.log("ERROR: timeout"); | ||
55 | phantom.exit(1); | ||
56 | }, 10000); | ||