blob: 6b8f6fec6166332253fcbfebc057a56c0e087e62 [file] [log] [blame]
Fathi Boudracd8f58d2012-06-15 17:43:39 +03001#!/usr/bin/python
2
Andy Doan15aa23b2012-02-02 16:40:58 -06003# Copyright (C) 2012 Linaro
4#
5# Author: Andy Doan <andy.doan@linaro.org>
6#
7# This file is part of Linaro Daily Prebuilt Images.
8#
9# Linaro Daily Prebuilt Images is free software; you can redistribute it and/or
10# modify it under the terms of the GNU General Public License
11# as published by the Free Software Foundation; either version 2
12# of the License, or (at your option) any later version.
13#
14# Linaro Daily Prebuilt Images is distributed in the hope that it will be useful,
15# but WITHOUT ANY WARRANTY; without even the implied warranty of
16# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17# GNU General Public License for more details.
18#
19# You should have received a copy of the GNU General Public License
20# along with Linaro Image Tools; if not, write to the Free Software
21# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
22# USA.
23
Andy Doanfccf4842012-03-08 10:42:19 -060024from linaro_fetch_image import fetch_image
Andy Doan15aa23b2012-02-02 16:40:58 -060025from linaro_image_tools import cmd_runner
26
Andy Doan8ff38f02012-02-20 16:32:37 -060027import crawler
28
Andy Doan15aa23b2012-02-02 16:40:58 -060029import argparse
Andy Doan15aa23b2012-02-02 16:40:58 -060030import datetime
Ricardo Salveti de Araujob39eb802012-06-23 00:55:31 -030031import time
Andy Doan3634c472012-04-17 12:50:58 -050032import hashlib
Andy Doanc6500af2012-03-15 14:44:39 -050033import os
Andy Doan15aa23b2012-02-02 16:40:58 -060034import re
Andy Doane449bce2012-03-21 15:22:31 -050035import urlparse
Andy Doan35ee9dc2012-05-11 01:10:12 -050036import string
Andy Doan15aa23b2012-02-02 16:40:58 -060037
Andy Doanea2e1ea2012-04-23 11:56:29 -050038class HwPack:
39 def __init__(self, lmcname, eula=False, pre_inst_script=None):
40 self.lmcname = lmcname
41 if eula:
42 self.eula = 'EULA.txt'
43 else:
Andy Doan4aabf022012-04-26 14:30:29 -050044 self.eula = 'OPEN-EULA.txt'
Andy Doanea2e1ea2012-04-23 11:56:29 -050045 self.pre_inst_script = pre_inst_script
46
47 def do_eula(self, imgfile):
48 fname = '%s/%s' % (os.path.dirname(imgfile), self.eula)
49 with file(fname, 'a'):
50 pass #just need the file to exist
51
Andy Doan15aa23b2012-02-02 16:40:58 -060052# a mapping of hwpack name to l-m-c name
53HWPACKS = {
Andy Doanea2e1ea2012-04-23 11:56:29 -050054 'beagleboard': HwPack('beagle'),
55 'overo': HwPack('overo'),
56 'lt-panda': HwPack('panda'),
Andy Doan7a8cc102012-04-26 09:25:33 -050057 'lt-panda-x11-base': HwPack('panda'),
Ricardo Salveti de Araujo90fe78d2012-07-29 00:51:57 -030058 'ti-panda-x11-base': HwPack('panda'),
Andy Doanea2e1ea2012-04-23 11:56:29 -050059 'lt-mx5': HwPack('mx53loco'),
60 'lt-mx6': HwPack('mx6qsabrelite'),
Andy Doan490f6202012-04-25 10:56:08 -050061 'lt-origen': HwPack('origen'),
62 'leb-origen': HwPack('origen'),
Andy Doanea2e1ea2012-04-23 11:56:29 -050063 'lt-snowball': HwPack('snowball_sd', True, 'ste-preinstall.sh'),
64 'lt-snowball-x11-base': HwPack('snowball_sd', True, 'ste-preinstall.sh'),
Ricardo Salveti de Araujoa798c502012-06-27 19:05:10 -030065 'lt-vexpress': HwPack('vexpress'),
Andy Doanedffb5d2012-04-25 11:22:19 -050066 'vexpress': HwPack('vexpress'),
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -030067 'vexpress64': HwPack('vexpress'),
Andy Doanea2e1ea2012-04-23 11:56:29 -050068 'efikamx': HwPack('efikamx'),
69 'igep': HwPack('igep'),
Andy Doan15aa23b2012-02-02 16:40:58 -060070}
71# a mapping of binary image to its image_file size
72BINARIES = {
Andy Doan4202d882012-03-29 21:36:09 -050073 # release images
Andy Doan08485da2012-05-31 22:24:29 +080074 'nano': '1G', #512 is big enough, but this gives us free disk space
Andy Doan4202d882012-03-29 21:36:09 -050075 'alip': '2G',
76 'ubuntu-desktop': '3G',
77 'linarotv-xbmc': '3G',
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -030078 'minimal-armv8': '512M',
Andy Doan15aa23b2012-02-02 16:40:58 -060079}
80
Fathi Boudra75e299a2012-08-26 19:04:03 +030081SNAPSHOTS_URL = "http://snapshots.linaro.org"
82
Andy Doan15aa23b2012-02-02 16:40:58 -060083def today():
84 d = datetime.date.today()
85 return "%d%02d%02d" % (d.year, d.month, d.day)
86
Ricardo Salveti de Araujob39eb802012-06-23 00:55:31 -030087def download(dm, url):
88 print "INFO: Fetching", url
89 before = time.time()
90 dfile = dm.download(url, None, verbose=False)
91 delta_min, delta_secs = divmod(int(time.time() - before), 60)
92 print "INFO: Downloaded file '%s' in %d min(s) and %d sec(s)" % (
93 dfile, delta_min, delta_secs)
94 return dfile
95
Andy Doan15aa23b2012-02-02 16:40:58 -060096def compress_image(imgfile):
Ricardo Salveti de Araujoba7dacf2012-06-23 01:08:24 -030097 print "INFO: compressing %s" % imgfile
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -030098 before = time.time()
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -030099 args = ('gzip', imgfile)
Andy Doan15aa23b2012-02-02 16:40:58 -0600100 cmd_runner.run(args).wait()
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300101 delta_min, delta_secs = divmod(int(time.time() - before), 60)
102 print "INFO: file compressed in %d min(s) and %d sec(s)" % (
103 delta_min, delta_secs)
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300104 return '%s.gz' % imgfile
Andy Doan23ce4e12012-03-21 12:46:21 -0500105
Ricardo Salveti de Araujod3743e92012-08-15 01:50:10 -0300106def zsync_image(platform, imgfile):
Ricardo Salveti de Araujoba7dacf2012-06-23 01:08:24 -0300107 print "INFO: making zsync file for %s" % imgfile
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300108 before = time.time()
Fathi Boudra75e299a2012-08-26 19:04:03 +0300109 basepath = "%s/%s/pre-built" % (SNAPSHOTS_URL, platform)
Paul Larson2a8e67b2012-08-06 09:47:14 -0500110 imgpath=os.sep.join(imgfile.split(os.sep)[-3:])
111 zurl=basepath + os.sep + imgpath
112 args = ('zsyncmake', '-b 2048', '-u', zurl, '-o', '%s.zsync' % imgfile,
113 imgfile)
Andy Doan23ce4e12012-03-21 12:46:21 -0500114 cmd_runner.run(args).wait()
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300115 delta_min, delta_secs = divmod(int(time.time() - before), 60)
116 print "INFO: zsync generated in %d min(s) and %d sec(s)" % (
117 delta_min, delta_secs)
Andy Doan15aa23b2012-02-02 16:40:58 -0600118
Andy Doan3634c472012-04-17 12:50:58 -0500119def md5sum(fname):
120 md5 = hashlib.md5()
121 with open(fname, 'rb') as f:
122 while True:
123 data = f.read(4096)
124 if not data: break
125 md5.update(data)
126 return md5.hexdigest()
127
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300128def create_image_info(infofile, gzfile, binary_file, hwpack_url, rootfs_url):
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300129 ''' prints out the md5sum and hwpack/rootfs info for the given image
Andy Doan3634c472012-04-17 12:50:58 -0500130 '''
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300131 print "INFO: building image info for: %s" % gzfile
132 md5 = md5sum(gzfile)
Andy Doan3634c472012-04-17 12:50:58 -0500133
134 binary_file = os.path.basename(binary_file)
Andy Doan35ee9dc2012-05-11 01:10:12 -0500135 tfile = '%s/image_info_template.html' % os.path.dirname(__file__)
Andy Doan3634c472012-04-17 12:50:58 -0500136
Andy Doan35ee9dc2012-05-11 01:10:12 -0500137 tmpl = None
138 with open(tfile) as f:
139 buff = f.read()
140 tmpl = string.Template(buff)
141
142 title = 'Pre-Built Image Info'
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300143
144 # build hwpack manifest and rootfs packages links
145 hwpack_re = re.search(r'(hwpack_linaro.*\d\w*)(.*)$', hwpack_url)
146 hwpack_name = "".join(hwpack_re.groups())
147 hwpack_manifest = hwpack_name + '.manifest.txt'
148 hwpack_manifest_url = re.sub('.tar.gz$', '.manifest.txt', hwpack_url)
149
150 rootfs_re = re.search(r'(linaro-.*\d)(.*)$', rootfs_url)
151 rootfs_name = "".join(rootfs_re.groups())
152 rootfs_packages = rootfs_name + '.packages'
153 rootfs_packages_url = re.sub('.tar.gz$', '.packages', rootfs_url)
154
Ricardo Salveti de Araujo894040a2012-05-29 14:07:55 -0300155 # try to get information from Jenkins, if available
156 jenkins_buildn = os.environ.get('BUILD_NUMBER') or 'Not Found'
157 jenkins_build_url = os.environ.get('BUILD_URL') or 'http://ci.linaro.org'
158
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300159 buff = tmpl.substitute(
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300160 title=title, md5=md5, image_name=os.path.basename(gzfile),
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300161 hwpack_name=hwpack_name, hwpack_url=hwpack_url,
162 rootfs_name=rootfs_name, rootfs_url=rootfs_url,
163 hwpack_manifest_name=hwpack_manifest,
164 hwpack_manifest_url=hwpack_manifest_url,
165 rootfs_packages_name=rootfs_packages,
Ricardo Salveti de Araujo894040a2012-05-29 14:07:55 -0300166 rootfs_packages_url=rootfs_packages_url,
167 jenkins_buildn=jenkins_buildn,
168 jenkins_build_url=jenkins_build_url)
Andy Doan3c0c3892012-04-25 11:09:53 -0500169 with open(infofile, 'w') as f:
Andy Doan35ee9dc2012-05-11 01:10:12 -0500170 f.write(buff)
Andy Doan3634c472012-04-17 12:50:58 -0500171
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300172def build_image(lmc, imgfile, hwpack, hwpack_file, binary,
173 binary_file, fastmodel=False):
Ricardo Salveti de Araujoba7dacf2012-06-23 01:08:24 -0300174 print "INFO: building image: %s" % imgfile
Andy Doanea2e1ea2012-04-23 11:56:29 -0500175 hwpi = HWPACKS[hwpack]
Andy Doan15aa23b2012-02-02 16:40:58 -0600176 size = BINARIES[binary]
177
Andy Doanc6500af2012-03-15 14:44:39 -0500178 args = [lmc,
Andy Doan15aa23b2012-02-02 16:40:58 -0600179 '--image_size', size,
180 '--hwpack-force-yes',
181 '--hwpack', hwpack_file,
182 '--binary', binary_file,
Andy Doanc6500af2012-03-15 14:44:39 -0500183 ]
184
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300185 if fastmodel:
186 args += ['--dev', 'fastmodel',
187 '--output-directory', os.path.dirname(imgfile),
188 '--image_file', os.path.basename(imgfile)]
189 else:
190 args += ['--dev', hwpi.lmcname,
191 '--image_file', imgfile]
192
Andy Doanea2e1ea2012-04-23 11:56:29 -0500193 if hwpi.pre_inst_script is not None:
Andy Doanc6500af2012-03-15 14:44:39 -0500194 sdir = os.path.abspath(os.path.dirname(__file__))
Andy Doanea2e1ea2012-04-23 11:56:29 -0500195 script = "%s/%s" % (sdir, hwpi.pre_inst_script)
Andy Doanc6500af2012-03-15 14:44:39 -0500196 print "using presintall script: %s" % script
197 args.append('--preinstall-script')
198 args.append(script)
199
Ricardo Salveti de Araujoba7dacf2012-06-23 01:08:24 -0300200 print "INFO: running l-m-c:", " ".join([str(arg) for arg in args])
201
Andy Doanea2e1ea2012-04-23 11:56:29 -0500202 hwpi.do_eula(imgfile)
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300203
204 # some useful information so we know how much time it took per lmc run
205 before = time.time()
Andy Doan15aa23b2012-02-02 16:40:58 -0600206 cmd_runner.run(args, as_root=True).wait()
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300207 delta_min, delta_secs = divmod(int(time.time() - before), 60)
208 print "INFO: linaro-media-create took %d min(s) and %d sec(s) to run" % (
209 delta_min, delta_secs)
Andy Doan15aa23b2012-02-02 16:40:58 -0600210
Andy Doan184cd3b2012-03-29 22:20:02 -0500211class SnapshotCrawler:
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300212 def __init__(self, platform, hwpacks, binaries):
Andy Doan184cd3b2012-03-29 22:20:02 -0500213 self.platform = platform
214 self.hwpacks = hwpacks
215 self.binaries = binaries
Andy Doane449bce2012-03-21 15:22:31 -0500216
Andy Doan184cd3b2012-03-29 22:20:02 -0500217 def get_binaries(self):
218 '''return a hash table of binary->url'''
219 binaryf = {}
220 for binary in self.binaries:
Fathi Boudra75e299a2012-08-26 19:04:03 +0300221 url = "%s/%s/images/%s" % (SNAPSHOTS_URL, self.platform, binary)
Andy Doan184cd3b2012-03-29 22:20:02 -0500222 (date, url) = crawler.latest_rfs(url)
Andy Doan4202d882012-03-29 21:36:09 -0500223 binaryf[binary] = url
Andy Doan4202d882012-03-29 21:36:09 -0500224
Andy Doan184cd3b2012-03-29 22:20:02 -0500225 return binaryf
Andy Doana3977372012-03-29 12:22:44 -0500226
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300227 def get_hwpacks(self):
228 '''return a hash table of hwpack->urls (date, url)'''
229 hwpackf = {}
230 for hwpack in self.hwpacks:
Fathi Boudra75e299a2012-08-26 19:04:03 +0300231 url = "%s/%s/hwpacks/%s" % (SNAPSHOTS_URL, self.platform, hwpack)
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300232 hwpackf[hwpack] = crawler.latest_hwpacks(url)
233
234 return hwpackf
Andy Doan4202d882012-03-29 21:36:09 -0500235
Andy Doan184cd3b2012-03-29 22:20:02 -0500236 def get_image_name(self, odir, hwpack, hwpack_url, binary):
237
238 #covert a url like:
Andy Doan540a4b12012-04-17 11:48:11 -0500239 # http://snapshots.linaro.org/precise/hwpacks/lt-panda/87/hwpack_linaro-lt-panda_20120417-87_armhf_supported.tar.gz
Andy Doan184cd3b2012-03-29 22:20:02 -0500240 # to:
Andy Doan66e2d5c2012-04-18 23:01:59 -0500241 # precise/pre-built/lt-panda/87/<file>.img
Andy Doan184cd3b2012-03-29 22:20:02 -0500242 path = urlparse.urlparse(hwpack_url).path
Andy Doan66e2d5c2012-04-18 23:01:59 -0500243 parts = path.split('/')
244 name = parts[-1] #use for the file name below
245 path = '/'.join(parts[-3:-1])
Andy Doan184cd3b2012-03-29 22:20:02 -0500246
247 path = '%s/%s' % (odir, path)
248 if not os.path.exists(path):
249 os.makedirs(path)
Andy Doan66e2d5c2012-04-18 23:01:59 -0500250
251 date = re.search(r'_(\d+-\d+)_', name).group(1)
Ricardo Salveti de Araujo99d52482012-06-22 16:05:09 -0300252 return "%s/%s-%s_%s_%s.img" % (path, hwpack, self.platform, binary, date)
Andy Doan184cd3b2012-03-29 22:20:02 -0500253
Andy Doan15aa23b2012-02-02 16:40:58 -0600254def main():
Andy Doan15aa23b2012-02-02 16:40:58 -0600255 hwpacks = HWPACKS.keys()
256 binaries = BINARIES.keys()
257
258 p = argparse.ArgumentParser(description=
Andy Doana3977372012-03-29 12:22:44 -0500259 'Builds a matrix of builds from the latest build on '
260 'snapshots.linaro.org or release images on releases.linaro.org')
Andy Doan15aa23b2012-02-02 16:40:58 -0600261
Andy Doan94bfe312012-02-09 17:57:29 -0800262 p.add_argument('-o', dest='out_dir', default='./out',
Andy Doan15aa23b2012-02-02 16:40:58 -0600263 help='The out directory for downloaded and built files, default=./')
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300264 p.add_argument('-d', dest='date', default="",
265 help='The date (YYYYMMDD) as reference for hwpack')
Andy Doan15aa23b2012-02-02 16:40:58 -0600266 p.add_argument('-w', dest='hwpacks', action='append',
267 help='The hwpacks to generate for, default=%s' %
268 ', '.join(hwpacks))
269 p.add_argument('-b', dest='binaries', action='append',
270 help='The binaries to generate for, default=%s' %
271 ', '.join(binaries))
Andy Doan540a4b12012-04-17 11:48:11 -0500272 p.add_argument('-p', dest='platform', default='precise',
273 help='The platform, default=precise')
Andy Doan4202d882012-03-29 21:36:09 -0500274 p.add_argument('-r', dest='release', action='store_true', default=False,
275 help='If this is for release images. NOTE: the "date" arg'
276 'will then become the cycle ie -d 12.03')
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300277 p.add_argument('-f', dest='fastmodel', action='store_true',
278 help='Build for the fastmodel target (for vexpress hwpack)')
279
Andy Doan15aa23b2012-02-02 16:40:58 -0600280
281 args = p.parse_args()
Andy Doana3977372012-03-29 12:22:44 -0500282
Andy Doan15aa23b2012-02-02 16:40:58 -0600283 if args.hwpacks:
284 hwpacks = args.hwpacks
285 if args.binaries:
286 binaries = args.binaries
287
Andy Doan1c6ef8b2012-04-23 12:28:50 -0500288 crawler.cookie_setup()
289
Andy Doan15aa23b2012-02-02 16:40:58 -0600290 dm = fetch_image.DownloadManager(args.out_dir)
Andy Doan15aa23b2012-02-02 16:40:58 -0600291
Andy Doan4202d882012-03-29 21:36:09 -0500292 if args.release:
Andy Doan184cd3b2012-03-29 22:20:02 -0500293 site = ReleaseCrawler(args.date, args.platform, hwpacks, binaries)
Andy Doan4202d882012-03-29 21:36:09 -0500294 else:
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300295 site = SnapshotCrawler(args.platform, hwpacks, binaries)
Andy Doan184cd3b2012-03-29 22:20:02 -0500296
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300297 hwpackf = site.get_hwpacks()
Andy Doan184cd3b2012-03-29 22:20:02 -0500298 binaryf = site.get_binaries()
Andy Doan15aa23b2012-02-02 16:40:58 -0600299
Andy Doanb00dbc02012-03-15 16:23:06 -0500300 lmc = '%s/linaro-image-tools/linaro-media-create' % args.out_dir
Andy Doan184cd3b2012-03-29 22:20:02 -0500301 odir = '%s/pre-built' % args.out_dir
Andy Doanb00dbc02012-03-15 16:23:06 -0500302
Andy Doan15aa23b2012-02-02 16:40:58 -0600303 for hwpack in hwpacks:
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300304 url = None
305 # check if the user wanted a hwpack from a specific date
306 if args.date:
307 try:
308 dates = [elem[0] for elem in hwpackf[hwpack]]
309 index = dates.index(args.date)
310 url = hwpackf[hwpack][index][1]
311 except:
312 print "ERROR: Could not find hwpack '%s' with date '%s'" \
313 ", skipping pre-built image for it" % (hwpack, args.date)
314 else:
315 # just grab the latest hwpack
316 try:
317 url = hwpackf[hwpack][0][1]
318 except:
319 print "ERROR: Could not find a valid hwpack url for '%s'" \
320 ", skipping pre-built image for it" % (hwpack)
Andy Doanb00dbc02012-03-15 16:23:06 -0500321
Andy Doan2cfe5e62012-02-09 17:41:45 -0800322 if url is not None and url is not False:
Ricardo Salveti de Araujob39eb802012-06-23 00:55:31 -0300323 hwpf = download(dm, url)
Andy Doan15aa23b2012-02-02 16:40:58 -0600324 for binary in binaries:
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300325 # create the image and metadata
Ricardo Salveti de Araujo99d52482012-06-22 16:05:09 -0300326 imgfile = site.get_image_name(odir, hwpack, url, binary)
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300327 infofile = '%s.html' % os.path.splitext(imgfile)[0]
Ricardo Salveti de Araujob39eb802012-06-23 00:55:31 -0300328 rfsf = download(dm, binaryf[binary])
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300329 build_image(lmc, imgfile, hwpack, hwpf, binary, rfsf,
330 args.fastmodel)
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300331 gzfile = compress_image(imgfile)
Ricardo Salveti de Araujod3743e92012-08-15 01:50:10 -0300332 zsync_image(args.platform, gzfile)
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300333 # TODO: Add support for OE based images
334 if args.platform != "openembedded":
335 create_image_info(infofile, gzfile,
336 imgfile, url, binaryf[binary])
Andy Doan15aa23b2012-02-02 16:40:58 -0600337
338if __name__ == '__main__':
339 main()
340