blob: 03513403add55ed20c6579c221faff44ecdf0b29 [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',
Ricardo Salveti de Araujod0fe5c42012-10-17 03:29:58 -030079 'lamp-armv8': '2G',
80 'sdk-armv8': '2G',
Andy Doan15aa23b2012-02-02 16:40:58 -060081}
82
Fathi Boudra75e299a2012-08-26 19:04:03 +030083SNAPSHOTS_URL = "http://snapshots.linaro.org"
84
Andy Doan15aa23b2012-02-02 16:40:58 -060085def today():
86 d = datetime.date.today()
87 return "%d%02d%02d" % (d.year, d.month, d.day)
88
Ricardo Salveti de Araujob39eb802012-06-23 00:55:31 -030089def download(dm, url):
90 print "INFO: Fetching", url
91 before = time.time()
92 dfile = dm.download(url, None, verbose=False)
93 delta_min, delta_secs = divmod(int(time.time() - before), 60)
94 print "INFO: Downloaded file '%s' in %d min(s) and %d sec(s)" % (
95 dfile, delta_min, delta_secs)
96 return dfile
97
Andy Doan15aa23b2012-02-02 16:40:58 -060098def compress_image(imgfile):
Ricardo Salveti de Araujoba7dacf2012-06-23 01:08:24 -030099 print "INFO: compressing %s" % imgfile
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300100 before = time.time()
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300101 args = ('gzip', imgfile)
Andy Doan15aa23b2012-02-02 16:40:58 -0600102 cmd_runner.run(args).wait()
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300103 delta_min, delta_secs = divmod(int(time.time() - before), 60)
104 print "INFO: file compressed in %d min(s) and %d sec(s)" % (
105 delta_min, delta_secs)
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300106 return '%s.gz' % imgfile
Andy Doan23ce4e12012-03-21 12:46:21 -0500107
Ricardo Salveti de Araujod3743e92012-08-15 01:50:10 -0300108def zsync_image(platform, imgfile):
Ricardo Salveti de Araujoba7dacf2012-06-23 01:08:24 -0300109 print "INFO: making zsync file for %s" % imgfile
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300110 before = time.time()
Fathi Boudra75e299a2012-08-26 19:04:03 +0300111 basepath = "%s/%s/pre-built" % (SNAPSHOTS_URL, platform)
Paul Larson2a8e67b2012-08-06 09:47:14 -0500112 imgpath=os.sep.join(imgfile.split(os.sep)[-3:])
113 zurl=basepath + os.sep + imgpath
114 args = ('zsyncmake', '-b 2048', '-u', zurl, '-o', '%s.zsync' % imgfile,
115 imgfile)
Andy Doan23ce4e12012-03-21 12:46:21 -0500116 cmd_runner.run(args).wait()
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300117 delta_min, delta_secs = divmod(int(time.time() - before), 60)
118 print "INFO: zsync generated in %d min(s) and %d sec(s)" % (
119 delta_min, delta_secs)
Andy Doan15aa23b2012-02-02 16:40:58 -0600120
Andy Doan3634c472012-04-17 12:50:58 -0500121def md5sum(fname):
122 md5 = hashlib.md5()
123 with open(fname, 'rb') as f:
124 while True:
125 data = f.read(4096)
126 if not data: break
127 md5.update(data)
128 return md5.hexdigest()
129
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300130def create_image_info(infofile, gzfile, binary_file, hwpack_url, rootfs_url):
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300131 ''' prints out the md5sum and hwpack/rootfs info for the given image
Andy Doan3634c472012-04-17 12:50:58 -0500132 '''
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300133 print "INFO: building image info for: %s" % gzfile
134 md5 = md5sum(gzfile)
Andy Doan3634c472012-04-17 12:50:58 -0500135
136 binary_file = os.path.basename(binary_file)
Andy Doan35ee9dc2012-05-11 01:10:12 -0500137 tfile = '%s/image_info_template.html' % os.path.dirname(__file__)
Andy Doan3634c472012-04-17 12:50:58 -0500138
Andy Doan35ee9dc2012-05-11 01:10:12 -0500139 tmpl = None
140 with open(tfile) as f:
141 buff = f.read()
142 tmpl = string.Template(buff)
143
144 title = 'Pre-Built Image Info'
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300145
146 # build hwpack manifest and rootfs packages links
147 hwpack_re = re.search(r'(hwpack_linaro.*\d\w*)(.*)$', hwpack_url)
148 hwpack_name = "".join(hwpack_re.groups())
149 hwpack_manifest = hwpack_name + '.manifest.txt'
150 hwpack_manifest_url = re.sub('.tar.gz$', '.manifest.txt', hwpack_url)
151
152 rootfs_re = re.search(r'(linaro-.*\d)(.*)$', rootfs_url)
153 rootfs_name = "".join(rootfs_re.groups())
154 rootfs_packages = rootfs_name + '.packages'
155 rootfs_packages_url = re.sub('.tar.gz$', '.packages', rootfs_url)
156
Ricardo Salveti de Araujo894040a2012-05-29 14:07:55 -0300157 # try to get information from Jenkins, if available
158 jenkins_buildn = os.environ.get('BUILD_NUMBER') or 'Not Found'
159 jenkins_build_url = os.environ.get('BUILD_URL') or 'http://ci.linaro.org'
160
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300161 buff = tmpl.substitute(
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300162 title=title, md5=md5, image_name=os.path.basename(gzfile),
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300163 hwpack_name=hwpack_name, hwpack_url=hwpack_url,
164 rootfs_name=rootfs_name, rootfs_url=rootfs_url,
165 hwpack_manifest_name=hwpack_manifest,
166 hwpack_manifest_url=hwpack_manifest_url,
167 rootfs_packages_name=rootfs_packages,
Ricardo Salveti de Araujo894040a2012-05-29 14:07:55 -0300168 rootfs_packages_url=rootfs_packages_url,
169 jenkins_buildn=jenkins_buildn,
170 jenkins_build_url=jenkins_build_url)
Andy Doan3c0c3892012-04-25 11:09:53 -0500171 with open(infofile, 'w') as f:
Andy Doan35ee9dc2012-05-11 01:10:12 -0500172 f.write(buff)
Andy Doan3634c472012-04-17 12:50:58 -0500173
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300174def build_image(lmc, imgfile, hwpack, hwpack_file, binary,
175 binary_file, fastmodel=False):
Ricardo Salveti de Araujoba7dacf2012-06-23 01:08:24 -0300176 print "INFO: building image: %s" % imgfile
Andy Doanea2e1ea2012-04-23 11:56:29 -0500177 hwpi = HWPACKS[hwpack]
Andy Doan15aa23b2012-02-02 16:40:58 -0600178 size = BINARIES[binary]
179
Andy Doanc6500af2012-03-15 14:44:39 -0500180 args = [lmc,
Andy Doan15aa23b2012-02-02 16:40:58 -0600181 '--image_size', size,
182 '--hwpack-force-yes',
183 '--hwpack', hwpack_file,
184 '--binary', binary_file,
Andy Doanc6500af2012-03-15 14:44:39 -0500185 ]
186
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300187 if fastmodel:
188 args += ['--dev', 'fastmodel',
189 '--output-directory', os.path.dirname(imgfile),
190 '--image_file', os.path.basename(imgfile)]
191 else:
192 args += ['--dev', hwpi.lmcname,
193 '--image_file', imgfile]
194
Andy Doanea2e1ea2012-04-23 11:56:29 -0500195 if hwpi.pre_inst_script is not None:
Andy Doanc6500af2012-03-15 14:44:39 -0500196 sdir = os.path.abspath(os.path.dirname(__file__))
Andy Doanea2e1ea2012-04-23 11:56:29 -0500197 script = "%s/%s" % (sdir, hwpi.pre_inst_script)
Andy Doanc6500af2012-03-15 14:44:39 -0500198 print "using presintall script: %s" % script
199 args.append('--preinstall-script')
200 args.append(script)
201
Ricardo Salveti de Araujoba7dacf2012-06-23 01:08:24 -0300202 print "INFO: running l-m-c:", " ".join([str(arg) for arg in args])
203
Andy Doanea2e1ea2012-04-23 11:56:29 -0500204 hwpi.do_eula(imgfile)
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300205
206 # some useful information so we know how much time it took per lmc run
207 before = time.time()
Andy Doan15aa23b2012-02-02 16:40:58 -0600208 cmd_runner.run(args, as_root=True).wait()
Ricardo Salveti de Araujod6788502012-06-23 01:19:18 -0300209 delta_min, delta_secs = divmod(int(time.time() - before), 60)
210 print "INFO: linaro-media-create took %d min(s) and %d sec(s) to run" % (
211 delta_min, delta_secs)
Andy Doan15aa23b2012-02-02 16:40:58 -0600212
Andy Doan184cd3b2012-03-29 22:20:02 -0500213class SnapshotCrawler:
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300214 def __init__(self, platform, hwpacks, binaries):
Andy Doan184cd3b2012-03-29 22:20:02 -0500215 self.platform = platform
216 self.hwpacks = hwpacks
217 self.binaries = binaries
Andy Doane449bce2012-03-21 15:22:31 -0500218
Andy Doan184cd3b2012-03-29 22:20:02 -0500219 def get_binaries(self):
220 '''return a hash table of binary->url'''
221 binaryf = {}
222 for binary in self.binaries:
Fathi Boudra75e299a2012-08-26 19:04:03 +0300223 url = "%s/%s/images/%s" % (SNAPSHOTS_URL, self.platform, binary)
Andy Doan184cd3b2012-03-29 22:20:02 -0500224 (date, url) = crawler.latest_rfs(url)
Andy Doan4202d882012-03-29 21:36:09 -0500225 binaryf[binary] = url
Andy Doan4202d882012-03-29 21:36:09 -0500226
Andy Doan184cd3b2012-03-29 22:20:02 -0500227 return binaryf
Andy Doana3977372012-03-29 12:22:44 -0500228
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300229 def get_hwpacks(self):
230 '''return a hash table of hwpack->urls (date, url)'''
231 hwpackf = {}
232 for hwpack in self.hwpacks:
Fathi Boudra75e299a2012-08-26 19:04:03 +0300233 url = "%s/%s/hwpacks/%s" % (SNAPSHOTS_URL, self.platform, hwpack)
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300234 hwpackf[hwpack] = crawler.latest_hwpacks(url)
235
236 return hwpackf
Andy Doan4202d882012-03-29 21:36:09 -0500237
Andy Doan184cd3b2012-03-29 22:20:02 -0500238 def get_image_name(self, odir, hwpack, hwpack_url, binary):
239
Fathi Boudra684e5e42012-11-30 04:19:33 +0100240 #convert a url like:
241 # http://snapshots.linaro.org/SERIE/hwpacks/HWPACK_NAME/HWPACK_BUILD_NUMBER/HWPACK_FILE_NAME
Andy Doan184cd3b2012-03-29 22:20:02 -0500242 # to:
Fathi Boudra684e5e42012-11-30 04:19:33 +0100243 # SERIE/pre-built/HWPACK_NAME/HWPACK_BUILD_NUMBER/HWPACK_FILE_NAME.img
Andy Doan184cd3b2012-03-29 22:20:02 -0500244 path = urlparse.urlparse(hwpack_url).path
Andy Doan66e2d5c2012-04-18 23:01:59 -0500245 parts = path.split('/')
246 name = parts[-1] #use for the file name below
247 path = '/'.join(parts[-3:-1])
Andy Doan184cd3b2012-03-29 22:20:02 -0500248
249 path = '%s/%s' % (odir, path)
250 if not os.path.exists(path):
251 os.makedirs(path)
Andy Doan66e2d5c2012-04-18 23:01:59 -0500252
253 date = re.search(r'_(\d+-\d+)_', name).group(1)
Ricardo Salveti de Araujo99d52482012-06-22 16:05:09 -0300254 return "%s/%s-%s_%s_%s.img" % (path, hwpack, self.platform, binary, date)
Andy Doan184cd3b2012-03-29 22:20:02 -0500255
Andy Doan15aa23b2012-02-02 16:40:58 -0600256def main():
Andy Doan15aa23b2012-02-02 16:40:58 -0600257 hwpacks = HWPACKS.keys()
258 binaries = BINARIES.keys()
259
260 p = argparse.ArgumentParser(description=
Andy Doana3977372012-03-29 12:22:44 -0500261 'Builds a matrix of builds from the latest build on '
262 'snapshots.linaro.org or release images on releases.linaro.org')
Andy Doan15aa23b2012-02-02 16:40:58 -0600263
Andy Doan94bfe312012-02-09 17:57:29 -0800264 p.add_argument('-o', dest='out_dir', default='./out',
Andy Doan15aa23b2012-02-02 16:40:58 -0600265 help='The out directory for downloaded and built files, default=./')
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300266 p.add_argument('-d', dest='date', default="",
267 help='The date (YYYYMMDD) as reference for hwpack')
Andy Doan15aa23b2012-02-02 16:40:58 -0600268 p.add_argument('-w', dest='hwpacks', action='append',
269 help='The hwpacks to generate for, default=%s' %
270 ', '.join(hwpacks))
271 p.add_argument('-b', dest='binaries', action='append',
272 help='The binaries to generate for, default=%s' %
273 ', '.join(binaries))
Fathi Boudra684e5e42012-11-30 04:19:33 +0100274 p.add_argument('-p', dest='platform', default='quantal',
275 help='The platform, default=quantal')
Andy Doan4202d882012-03-29 21:36:09 -0500276 p.add_argument('-r', dest='release', action='store_true', default=False,
277 help='If this is for release images. NOTE: the "date" arg'
278 'will then become the cycle ie -d 12.03')
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300279 p.add_argument('-f', dest='fastmodel', action='store_true',
280 help='Build for the fastmodel target (for vexpress hwpack)')
281
Andy Doan15aa23b2012-02-02 16:40:58 -0600282
283 args = p.parse_args()
Andy Doana3977372012-03-29 12:22:44 -0500284
Andy Doan15aa23b2012-02-02 16:40:58 -0600285 if args.hwpacks:
286 hwpacks = args.hwpacks
287 if args.binaries:
288 binaries = args.binaries
289
Andy Doan1c6ef8b2012-04-23 12:28:50 -0500290 crawler.cookie_setup()
291
Andy Doan15aa23b2012-02-02 16:40:58 -0600292 dm = fetch_image.DownloadManager(args.out_dir)
Andy Doan15aa23b2012-02-02 16:40:58 -0600293
Andy Doan4202d882012-03-29 21:36:09 -0500294 if args.release:
Andy Doan184cd3b2012-03-29 22:20:02 -0500295 site = ReleaseCrawler(args.date, args.platform, hwpacks, binaries)
Andy Doan4202d882012-03-29 21:36:09 -0500296 else:
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300297 site = SnapshotCrawler(args.platform, hwpacks, binaries)
Andy Doan184cd3b2012-03-29 22:20:02 -0500298
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300299 hwpackf = site.get_hwpacks()
Andy Doan184cd3b2012-03-29 22:20:02 -0500300 binaryf = site.get_binaries()
Andy Doan15aa23b2012-02-02 16:40:58 -0600301
Andy Doanb00dbc02012-03-15 16:23:06 -0500302 lmc = '%s/linaro-image-tools/linaro-media-create' % args.out_dir
Andy Doan184cd3b2012-03-29 22:20:02 -0500303 odir = '%s/pre-built' % args.out_dir
Andy Doanb00dbc02012-03-15 16:23:06 -0500304
Andy Doan15aa23b2012-02-02 16:40:58 -0600305 for hwpack in hwpacks:
Ricardo Salveti de Araujo0352cb92012-06-22 21:48:34 -0300306 url = None
307 # check if the user wanted a hwpack from a specific date
308 if args.date:
309 try:
310 dates = [elem[0] for elem in hwpackf[hwpack]]
311 index = dates.index(args.date)
312 url = hwpackf[hwpack][index][1]
313 except:
314 print "ERROR: Could not find hwpack '%s' with date '%s'" \
315 ", skipping pre-built image for it" % (hwpack, args.date)
316 else:
317 # just grab the latest hwpack
318 try:
319 url = hwpackf[hwpack][0][1]
320 except:
321 print "ERROR: Could not find a valid hwpack url for '%s'" \
322 ", skipping pre-built image for it" % (hwpack)
Andy Doanb00dbc02012-03-15 16:23:06 -0500323
Andy Doan2cfe5e62012-02-09 17:41:45 -0800324 if url is not None and url is not False:
Ricardo Salveti de Araujob39eb802012-06-23 00:55:31 -0300325 hwpf = download(dm, url)
Andy Doan15aa23b2012-02-02 16:40:58 -0600326 for binary in binaries:
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300327 # create the image and metadata
Ricardo Salveti de Araujo99d52482012-06-22 16:05:09 -0300328 imgfile = site.get_image_name(odir, hwpack, url, binary)
Ricardo Salveti de Araujob93a7ec2012-05-29 14:07:53 -0300329 infofile = '%s.html' % os.path.splitext(imgfile)[0]
Ricardo Salveti de Araujob39eb802012-06-23 00:55:31 -0300330 rfsf = download(dm, binaryf[binary])
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300331 build_image(lmc, imgfile, hwpack, hwpf, binary, rfsf,
332 args.fastmodel)
Ricardo Salveti de Araujo7edc48b2012-06-23 03:55:04 -0300333 gzfile = compress_image(imgfile)
Ricardo Salveti de Araujod3743e92012-08-15 01:50:10 -0300334 zsync_image(args.platform, gzfile)
Ricardo Salveti de Araujo688f24f2012-10-17 02:40:36 -0300335 # TODO: Add support for OE based images
336 if args.platform != "openembedded":
337 create_image_info(infofile, gzfile,
338 imgfile, url, binaryf[binary])
Andy Doan15aa23b2012-02-02 16:40:58 -0600339
340if __name__ == '__main__':
341 main()
342