1# Copyright (C) 2009 Google Inc. All rights reserved. 2# 3# Redistribution and use in source and binary forms, with or without 4# modification, are permitted provided that the following conditions are 5# met: 6# 7# * Redistributions of source code must retain the above copyright 8# notice, this list of conditions and the following disclaimer. 9# * Redistributions in binary form must reproduce the above 10# copyright notice, this list of conditions and the following disclaimer 11# in the documentation and/or other materials provided with the 12# distribution. 13# * Neither the name of Google Inc. nor the names of its 14# contributors may be used to endorse or promote products derived from 15# this software without specific prior written permission. 16# 17# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 18# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 19# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 20# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 21# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 22# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 23# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 24# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 25# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 26# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 27# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 28 29from webkitpy.networktransaction import NetworkTransaction 30from webkitpy.webkit_logging import log 31from mechanize import Browser 32 33# WebKit includes a built copy of BeautifulSoup in Scripts/webkitpy 34# so this import should always succeed. 35from .BeautifulSoup import BeautifulSoup 36 37import urllib2 38 39 40class StatusServer: 41 default_host = "webkit-commit-queue.appspot.com" 42 43 def __init__(self, host=default_host): 44 self.set_host(host) 45 self.browser = Browser() 46 47 def set_host(self, host): 48 self.host = host 49 self.url = "http://%s" % self.host 50 51 def results_url_for_status(self, status_id): 52 return "%s/results/%s" % (self.url, status_id) 53 54 def _add_patch(self, patch): 55 if not patch: 56 return 57 if patch.bug_id(): 58 self.browser["bug_id"] = str(patch.bug_id()) 59 if patch.id(): 60 self.browser["patch_id"] = str(patch.id()) 61 62 def _add_results_file(self, results_file): 63 if not results_file: 64 return 65 self.browser.add_file(results_file, "text/plain", "results.txt", 'results_file') 66 67 def _post_to_server(self, queue_name, status, patch, results_file): 68 if results_file: 69 # We might need to re-wind the file if we've already tried to post it. 70 results_file.seek(0) 71 72 update_status_url = "%s/update-status" % self.url 73 self.browser.open(update_status_url) 74 self.browser.select_form(name="update_status") 75 self.browser['queue_name'] = queue_name 76 self._add_patch(patch) 77 self.browser['status'] = status 78 self._add_results_file(results_file) 79 return self.browser.submit().read() # This is the id of the newly created status object. 80 81 def update_status(self, queue_name, status, patch=None, results_file=None): 82 # During unit testing, host is None 83 if not self.host: 84 return 85 86 log(status) 87 return NetworkTransaction().run(lambda: self._post_to_server(queue_name, status, patch, results_file)) 88 89 def patch_status(self, queue_name, patch_id): 90 update_status_url = "%s/patch-status/%s/%s" % (self.url, queue_name, patch_id) 91 try: 92 return urllib2.urlopen(update_status_url).read() 93 except urllib2.HTTPError, e: 94 if e.code == 404: 95 return None 96 raise e 97