blob: 240e3a2fe795e9249b9831bf14194229ddce1f69 [file] [log] [blame]
rtc@google.comded22402009-10-26 22:36:21 +00001# Copyright (c) 2009 The Chromium OS 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
rtc@google.com64244662009-11-12 00:52:08 +00005from buildutil import BuildObject
rtc@google.comded22402009-10-26 22:36:21 +00006from xml.dom import minidom
7
8import os
Darin Petkov798fe7d2010-03-22 15:18:13 -07009import shutil
rtc@google.comded22402009-10-26 22:36:21 +000010import web
11
rtc@google.com64244662009-11-12 00:52:08 +000012class Autoupdate(BuildObject):
Darin Petkov798fe7d2010-03-22 15:18:13 -070013 # Basic functionality of handling ChromeOS autoupdate pings
rtc@google.com21a5ca32009-11-04 18:23:23 +000014 # and building/serving update images.
15 # TODO(rtc): Clean this code up and write some tests.
rtc@google.comded22402009-10-26 22:36:21 +000016
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070017 def __init__(self, serve_only=None, test_image=False, *args, **kwargs):
18 self.serve_only = serve_only
19 if serve_only:
20 web.debug('Autoupdate in "serve update images only" mode.')
21 self.test_image=test_image
22 super(Autoupdate, self).__init__(*args, **kwargs)
23
rtc@google.com21a5ca32009-11-04 18:23:23 +000024 def GetUpdatePayload(self, hash, size, url):
25 payload = """<?xml version="1.0" encoding="UTF-8"?>
26 <gupdate xmlns="http://www.google.com/update2/response" protocol="2.0">
27 <app appid="{%s}" status="ok">
28 <ping status="ok"/>
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070029 <updatecheck
30 codebase="%s"
31 hash="%s"
32 needsadmin="false"
33 size="%s"
rtc@google.com21a5ca32009-11-04 18:23:23 +000034 status="ok"/>
35 </app>
36 </gupdate>
37 """
38 return payload % (self.app_id, url, hash, size)
rtc@google.comded22402009-10-26 22:36:21 +000039
rtc@google.com21a5ca32009-11-04 18:23:23 +000040 def GetNoUpdatePayload(self):
41 payload = """<?xml version="1.0" encoding="UTF-8"?>
42 <gupdate xmlns="http://www.google.com/update2/response" protocol="2.0">
43 <app appid="{%s}" status="ok">
44 <ping status="ok"/>
45 <updatecheck status="noupdate"/>
46 </app>
47 </gupdate>
48 """
49 return payload % self.app_id
rtc@google.comded22402009-10-26 22:36:21 +000050
Sam Leffler76382042010-02-18 09:58:42 -080051 def GetLatestImagePath(self, board_id):
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070052 cmd = '%s/get_latest_image.sh --board %s' % (self.scripts_dir, board_id)
rtc@google.com21a5ca32009-11-04 18:23:23 +000053 return os.popen(cmd).read().strip()
rtc@google.comded22402009-10-26 22:36:21 +000054
rtc@google.com21a5ca32009-11-04 18:23:23 +000055 def GetLatestVersion(self, latest_image_path):
56 latest_version = latest_image_path.split('/')[-1]
Ryan Cairns1b05beb2010-02-05 17:05:24 -080057
58 # Removes the portage build prefix.
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070059 latest_version = latest_version.lstrip('g-')
rtc@google.com21a5ca32009-11-04 18:23:23 +000060 return latest_version.split('-')[0]
rtc@google.comded22402009-10-26 22:36:21 +000061
rtc@google.com21a5ca32009-11-04 18:23:23 +000062 def CanUpdate(self, client_version, latest_version):
63 """
64 Returns true iff the latest_version is greater than the client_version.
65 """
66 client_tokens = client_version.split('.')
67 latest_tokens = latest_version.split('.')
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070068 web.debug('client version %s latest version %s' \
Charlie Lee8c993082010-02-24 13:27:37 -080069 % (client_version, latest_version))
rtc@google.com21a5ca32009-11-04 18:23:23 +000070 for i in range(0,4):
71 if int(latest_tokens[i]) == int(client_tokens[i]):
72 continue
73 return int(latest_tokens[i]) > int(client_tokens[i])
rtc@google.comded22402009-10-26 22:36:21 +000074 return False
rtc@google.comded22402009-10-26 22:36:21 +000075
rtc@google.com21a5ca32009-11-04 18:23:23 +000076 def BuildUpdateImage(self, image_path):
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070077 if self.test_image:
78 image_file = '%s/rootfs_test.image' % image_path
79 else:
80 image_file = '%s/rootfs.image' % image_path
81 update_file = '%s/update.gz' % image_path
82 if (os.path.exists(update_file) and
83 os.path.getmtime(update_file) >= os.path.getmtime(image_file)):
84 web.debug('Found cached update image %s/update.gz' % image_path)
85 else:
86 web.debug('generating update image %s/update.gz' % image_path)
87 mkupdate = '%s/mk_memento_images.sh %s' % (self.scripts_dir, image_file)
rtc@google.com21a5ca32009-11-04 18:23:23 +000088 web.debug(mkupdate)
89 err = os.system(mkupdate)
90 if err != 0:
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070091 web.debug('failed to create update image')
rtc@google.com21a5ca32009-11-04 18:23:23 +000092 return False
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070093 if not self.serve_only:
94 web.debug('Found an image, copying it to static')
95 try:
Darin Petkov798fe7d2010-03-22 15:18:13 -070096 shutil.copy('%s/update.gz' % image_path, self.static_dir)
Sean O'Connor14b6a0a2010-03-20 23:23:48 -070097 except Exception, e:
98 web.debug('Unable to copy update.gz from %s to %s' \
99 % (image_path, self.static_dir))
100 return False
rtc@google.com21a5ca32009-11-04 18:23:23 +0000101 return True
rtc@google.comded22402009-10-26 22:36:21 +0000102
rtc@google.com21a5ca32009-11-04 18:23:23 +0000103 def GetSize(self, update_path):
104 return os.path.getsize(update_path)
rtc@google.comded22402009-10-26 22:36:21 +0000105
rtc@google.com21a5ca32009-11-04 18:23:23 +0000106 def GetHash(self, update_path):
Darin Petkov8ef83452010-03-23 16:52:29 -0700107 cmd = "cat %s | openssl sha1 -binary | openssl base64 | tr \'\\n\' \' \';" \
108 % update_path
109 web.debug(cmd)
110 return os.popen(cmd).read()
111
rtc@google.comded22402009-10-26 22:36:21 +0000112
Sean O'Connor14b6a0a2010-03-20 23:23:48 -0700113 def HandleUpdatePing(self, data, label=None):
rtc@google.com21a5ca32009-11-04 18:23:23 +0000114 update_dom = minidom.parseString(data)
115 root = update_dom.firstChild
Sean O'Connor14b6a0a2010-03-20 23:23:48 -0700116 query = root.getElementsByTagName('o:app')[0]
Charlie Lee8c993082010-02-24 13:27:37 -0800117 client_version = query.getAttribute('version')
118 board_id = query.hasAttribute('board') and query.getAttribute('board') \
Sean O'Connor14b6a0a2010-03-20 23:23:48 -0700119 or 'x86-generic'
Charlie Lee8c993082010-02-24 13:27:37 -0800120 latest_image_path = self.GetLatestImagePath(board_id)
121 latest_version = self.GetLatestVersion(latest_image_path)
Sean O'Connor14b6a0a2010-03-20 23:23:48 -0700122 if client_version != 'ForcedUpdate' \
Charlie Lee8c993082010-02-24 13:27:37 -0800123 and not self.CanUpdate(client_version, latest_version):
Sean O'Connor14b6a0a2010-03-20 23:23:48 -0700124 web.debug('no update')
rtc@google.com21a5ca32009-11-04 18:23:23 +0000125 return self.GetNoUpdatePayload()
rtc@google.com21a5ca32009-11-04 18:23:23 +0000126 hostname = web.ctx.host
Sean O'Connor14b6a0a2010-03-20 23:23:48 -0700127 if label:
128 web.debug('Client requested version %s' % label)
129 # Check that matching build exists
130 image_path = '%s/%s' % (self.static_dir, label)
131 if not os.path.exists(image_path):
132 web.debug('%s not found.' % image_path)
133 return self.GetNoUpdatePayload()
134 # Construct a response
135 ok = self.BuildUpdateImage(image_path)
136 if ok != True:
137 web.debug('Failed to build an update image')
138 return self.GetNoUpdatePayload()
139 web.debug('serving update: ')
140 hash = self.GetHash('%s/%s/update.gz' % (self.static_dir, label))
141 size = self.GetSize('%s/%s/update.gz' % (self.static_dir, label))
142 url = 'http://%s/static/archive/%s/update.gz' % (hostname, label)
143 return self.GetUpdatePayload(hash, size, url)
144 web.debug( 'DONE')
145 else:
146 web.debug('update found %s ' % latest_version)
147 ok = self.BuildUpdateImage(latest_image_path)
148 if ok != True:
149 web.debug('Failed to build an update image')
150 return self.GetNoUpdatePayload()
rtc@google.comded22402009-10-26 22:36:21 +0000151
Sean O'Connor14b6a0a2010-03-20 23:23:48 -0700152 hash = self.GetHash('%s/update.gz' % self.static_dir)
153 size = self.GetSize('%s/update.gz' % self.static_dir)
154
155 url = 'http://%s/static/update.gz' % hostname
156 return self.GetUpdatePayload(hash, size, url)