# Copyright (c) 2013 The Chromium OS Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. import BaseHTTPServer import thread import urlparse from autotest_lib.client.bin import test, utils from autotest_lib.client.common_lib import error, utils def _split_url(url): """Splits a URL into the URL base and path.""" split_url = urlparse.urlsplit(url) url_base = urlparse.urlunsplit( (split_url.scheme, split_url.netloc, '', '', '')) url_path = split_url.path return url_base, url_path.lstrip('/') class NanoOmahaDevserver(object): """Simple implementation of Omaha.""" class Handler(BaseHTTPServer.BaseHTTPRequestHandler): """Inner class for handling HTTP requests.""" _OMAHA_RESPONSE_TEMPLATE_HEAD = """ <response protocol=\"3.0\"> <daystart elapsed_seconds=\"44801\"/> <app appid=\"{87efface-864d-49a5-9bb3-4b050a7c227a}\" status=\"ok\"> <ping status=\"ok\"/> <updatecheck status=\"ok\"> <urls> <url codebase=\"%s\"/> </urls> <manifest version=\"9999.0.0\"> <packages> <package name=\"%s\" size=\"%d\" required=\"true\"/> </packages> <actions> <action event=\"postinstall\" ChromeOSVersion=\"9999.0.0\" sha256=\"%s\" needsadmin=\"false\" IsDeltaPayload=\"false\" """ _OMAHA_RESPONSE_TEMPLATE_TAIL = """ /> </actions> </manifest> </updatecheck> </app> </response> """ def do_POST(self): """Handler for POST requests.""" if self.path == '/update': (base, name) = _split_url(self.server._devserver._image_url) response = self._OMAHA_RESPONSE_TEMPLATE_HEAD % ( base + '/', name, self.server._devserver._image_size, self.server._devserver._sha256) if self.server._devserver._metadata_size: response += ' MetadataSize="%d"\n' % ( self.server._devserver._metadata_size) if self.server._devserver._metadata_signature: response += ' MetadataSignatureRsa="%s"\n' % ( self.server._devserver._metadata_signature) if self.server._devserver._public_key: response += ' PublicKeyRsa="%s"\n' % ( self.server._devserver._public_key) response += self._OMAHA_RESPONSE_TEMPLATE_TAIL self.send_response(200) self.send_header('Content-Type', 'application/xml') self.end_headers() self.wfile.write(response) else: self.send_response(500) def start(self): """Starts the server.""" self._httpd = BaseHTTPServer.HTTPServer(('127.0.0.1', 0), self.Handler) self._httpd._devserver = self # Serve HTTP requests in a dedicated thread. thread.start_new_thread(self._httpd.serve_forever, ()) self._port = self._httpd.socket.getsockname()[1] def stop(self): """Stops the server.""" self._httpd.shutdown() def get_port(self): """Returns the TCP port number the server is listening on.""" return self._port def set_image_params(self, image_url, image_size, sha256, metadata_size=None, metadata_signature=None, public_key=None): """Sets the values to return in the Omaha response. Only the |image_url|, |image_size| and |sha256| parameters are mandatory.""" self._image_url = image_url self._image_size = image_size self._sha256 = sha256 self._metadata_size = metadata_size self._metadata_signature = metadata_signature self._public_key = public_key class autoupdate_CannedOmahaUpdate(test.test): """Client-side mechanism to update a DUT with a given image.""" version = 1 """Restarts update_engine and attempts an update from the image pointed to by |image_url| of size |image_size| with checksum |image_sha256|. The |metadata_size|, |metadata_signature| and |public_key| parameters are optional. If the |allow_failure| parameter is True, then the test will succeed even if the update failed.""" def run_once(self, image_url, image_size, image_sha256, allow_failure=False, metadata_size=None, metadata_signature=None, public_key=None): utils.run('restart update-engine') omaha = NanoOmahaDevserver() omaha.set_image_params(image_url, image_size, image_sha256, metadata_size, metadata_signature, public_key) omaha.start() try: utils.run('update_engine_client -omaha_url=' + 'http://127.0.0.1:%d/update ' % omaha.get_port() + '-update') except error.CmdError: omaha.stop() if not allow_failure: raise error.TestFail('Update attempt failed.')