epoger@google.com | fd04011 | 2013-08-20 16:21:55 +0000 | [diff] [blame] | 1 | #!/usr/bin/python |
| 2 | |
| 3 | # Copyright (c) 2013 The Chromium Authors. All rights reserved. |
| 4 | # Use of this source code is governed by a BSD-style license that can be |
| 5 | # found in the LICENSE file. |
| 6 | |
| 7 | """ |
| 8 | Provides read access to buildbot's global_variables.json . |
| 9 | """ |
| 10 | |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 11 | |
| 12 | from contextlib import closing |
| 13 | |
| 14 | import HTMLParser |
epoger@google.com | fd04011 | 2013-08-20 16:21:55 +0000 | [diff] [blame] | 15 | import json |
commit-bot@chromium.org | 9f8e282 | 2014-01-14 19:18:45 +0000 | [diff] [blame] | 16 | import re |
epoger@google.com | fd04011 | 2013-08-20 16:21:55 +0000 | [diff] [blame] | 17 | import svn |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 18 | import sys |
| 19 | import urllib2 |
| 20 | |
epoger@google.com | fd04011 | 2013-08-20 16:21:55 +0000 | [diff] [blame] | 21 | |
| 22 | _global_vars = None |
| 23 | |
borenet@google.com | 4b897fa | 2013-12-02 20:27:16 +0000 | [diff] [blame] | 24 | |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 25 | GLOBAL_VARS_JSON_URL = ('https://skia.googlesource.com/buildbot/+' |
| 26 | '/master/site_config/global_variables.json') |
borenet@google.com | 4b897fa | 2013-12-02 20:27:16 +0000 | [diff] [blame] | 27 | |
| 28 | |
| 29 | class GlobalVarsRetrievalError(Exception): |
| 30 | """Exception which is raised when the global_variables.json file cannot be |
| 31 | retrieved from the Skia buildbot repository.""" |
epoger@google.com | fd04011 | 2013-08-20 16:21:55 +0000 | [diff] [blame] | 32 | pass |
| 33 | |
borenet@google.com | 4b897fa | 2013-12-02 20:27:16 +0000 | [diff] [blame] | 34 | |
| 35 | class JsonDecodeError(Exception): |
| 36 | """Exception which is raised when the global_variables.json file cannot be |
| 37 | interpreted as JSON. This may be due to the file itself being incorrectly |
| 38 | formatted or due to an incomplete or corrupted downloaded version of the file. |
| 39 | """ |
| 40 | pass |
| 41 | |
| 42 | |
| 43 | class NoSuchGlobalVariable(KeyError): |
| 44 | """Exception which is raised when a given variable is not found in the |
| 45 | global_variables.json file.""" |
| 46 | pass |
| 47 | |
| 48 | |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 49 | def retrieve_from_googlesource(url): |
| 50 | """Retrieve the given file from GoogleSource's HTTP interface, trimming the |
| 51 | extraneous HTML. Intended to be a GoogleSource equivalent of "svn cat". |
| 52 | |
| 53 | This just returns the unescaped contents of the first <pre> tag which matches |
| 54 | our expectations for GoogleSource's HTTP interface. If that interface changes, |
| 55 | this function will almost surely break. |
| 56 | |
| 57 | Args: |
| 58 | url: string; the URL of the file to retrieve. |
| 59 | Returns: |
| 60 | The contents of the file in GoogleSource, stripped of the extra HTML from |
| 61 | the HTML interface. |
| 62 | """ |
| 63 | with closing(urllib2.urlopen(url)) as f: |
| 64 | contents = f.read() |
commit-bot@chromium.org | 9f8e282 | 2014-01-14 19:18:45 +0000 | [diff] [blame] | 65 | pre_open = '<pre class="git-blob prettyprint linenums lang-(\w+)">' |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 66 | pre_close = '</pre>' |
commit-bot@chromium.org | 9f8e282 | 2014-01-14 19:18:45 +0000 | [diff] [blame] | 67 | matched_tag = re.search(pre_open, contents).group() |
| 68 | start_index = contents.find(matched_tag) |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 69 | end_index = contents.find(pre_close) |
| 70 | parser = HTMLParser.HTMLParser() |
commit-bot@chromium.org | 9f8e282 | 2014-01-14 19:18:45 +0000 | [diff] [blame] | 71 | return parser.unescape(contents[start_index + len(matched_tag):end_index]) |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 72 | |
| 73 | |
epoger@google.com | fd04011 | 2013-08-20 16:21:55 +0000 | [diff] [blame] | 74 | def Get(var_name): |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 75 | """Return the value associated with this name in global_variables.json. |
| 76 | |
| 77 | Args: |
| 78 | var_name: string; the variable to look up. |
| 79 | Returns: |
| 80 | The value of the variable. |
| 81 | Raises: |
| 82 | NoSuchGlobalVariable if there is no variable with that name. |
| 83 | """ |
epoger@google.com | fd04011 | 2013-08-20 16:21:55 +0000 | [diff] [blame] | 84 | global _global_vars |
| 85 | if not _global_vars: |
borenet@google.com | 4b897fa | 2013-12-02 20:27:16 +0000 | [diff] [blame] | 86 | try: |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 87 | global_vars_text = retrieve_from_googlesource(GLOBAL_VARS_JSON_URL) |
borenet@google.com | 4b897fa | 2013-12-02 20:27:16 +0000 | [diff] [blame] | 88 | except Exception: |
| 89 | raise GlobalVarsRetrievalError('Failed to retrieve %s.' % |
| 90 | GLOBAL_VARS_JSON_URL) |
| 91 | try: |
| 92 | _global_vars = json.loads(global_vars_text) |
| 93 | except ValueError as e: |
| 94 | raise JsonDecodeError(e.message + '\n' + global_vars_text) |
epoger@google.com | fd04011 | 2013-08-20 16:21:55 +0000 | [diff] [blame] | 95 | try: |
| 96 | return _global_vars[var_name]['value'] |
| 97 | except KeyError: |
| 98 | raise NoSuchGlobalVariable(var_name) |
borenet@google.com | 6f0f5b4 | 2014-01-09 21:41:39 +0000 | [diff] [blame] | 99 | |
| 100 | |
| 101 | if __name__ == '__main__': |
| 102 | print Get(sys.argv[1]) |