|  | # | 
|  | # Copyright (C) 2015 The Android Open Source Project | 
|  | # | 
|  | # Licensed under the Apache License, Version 2.0 (the 'License'); | 
|  | # you may not use this file except in compliance with the License. | 
|  | # You may obtain a copy of the License at | 
|  | # | 
|  | #      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | # | 
|  | # Unless required by applicable law or agreed to in writing, software | 
|  | # distributed under the License is distributed on an 'AS IS' BASIS, | 
|  | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | # See the License for the specific language governing permissions and | 
|  | # limitations under the License. | 
|  | # | 
|  | import httplib | 
|  | import httplib2 | 
|  | import logging | 
|  | import re | 
|  | import socket | 
|  |  | 
|  | import apiclient.errors | 
|  |  | 
|  | import gerrit | 
|  | import gmail | 
|  | import presubmit | 
|  |  | 
|  |  | 
|  | def get_gerrit_info(body): | 
|  | info = {} | 
|  | gerrit_pattern = r'^Gerrit-(\S+): (.+)$' | 
|  | for match in re.finditer(gerrit_pattern, body, flags=re.MULTILINE): | 
|  | info[match.group(1)] = match.group(2).strip() | 
|  | return info | 
|  |  | 
|  |  | 
|  | def process_message(msg, dry_run): | 
|  | try: | 
|  | body = gmail.get_body(msg) | 
|  | gerrit_info = get_gerrit_info(body) | 
|  | if not gerrit_info: | 
|  | logging.fatal('No Gerrit info found: %s', msg.subject) | 
|  | msg_type = gerrit_info['MessageType'] | 
|  | handlers = { | 
|  | 'comment': presubmit.handle_comment, | 
|  | 'newchange': presubmit.handle_change, | 
|  | 'newpatchset': presubmit.handle_change, | 
|  |  | 
|  | 'abandon': presubmit.skip_handler, | 
|  | 'merge-failed': presubmit.skip_handler, | 
|  | 'merged': presubmit.skip_handler, | 
|  | 'restore': presubmit.skip_handler, | 
|  | 'revert': presubmit.skip_handler, | 
|  | } | 
|  |  | 
|  | message_type = gerrit_info['MessageType'] | 
|  | if message_type in handlers: | 
|  | return handlers[message_type](gerrit_info, body, dry_run) | 
|  | else: | 
|  | logging.warning('MessageType %s unhandled.', msg_type) | 
|  | return False | 
|  | except NotImplementedError as ex: | 
|  | logging.error("%s", ex) | 
|  | return False | 
|  | except gerrit.GerritError as ex: | 
|  | change_id = gerrit_info['Change-Id'] | 
|  | logging.error('Gerrit error (%d): %s %s', ex.code, change_id, ex.url) | 
|  | return ex.code == 404 | 
|  |  | 
|  |  | 
|  | def get_and_process_jobs(): | 
|  | dry_run = False | 
|  |  | 
|  | gmail_service = gmail.build_service() | 
|  | msg_service = gmail_service.users().messages() | 
|  |  | 
|  | # We run in a loop because some of the exceptions thrown here mean we just | 
|  | # need to retry. For errors where we should back off (typically any gmail | 
|  | # API exceptions), process_changes catches the error and returns normally. | 
|  | while True: | 
|  | try: | 
|  | process_changes(gmail_service, msg_service, dry_run) | 
|  | return | 
|  | except httplib.BadStatusLine: | 
|  | pass | 
|  | except httplib2.ServerNotFoundError: | 
|  | pass | 
|  | except socket.error: | 
|  | pass | 
|  |  | 
|  |  | 
|  | def process_changes(gmail_service, msg_service, dry_run): | 
|  | try: | 
|  | labels = gmail_service.users().labels().list(userId='me').execute() | 
|  | if not labels['labels']: | 
|  | logging.error('Could not retrieve Gmail labels') | 
|  | return | 
|  | label_id = gmail.get_gerrit_label(labels['labels']) | 
|  | if not label_id: | 
|  | logging.error('Could not find gerrit label') | 
|  | return | 
|  |  | 
|  | for msg in gmail.get_all_messages(gmail_service, label_id): | 
|  | msg = msg_service.get(userId='me', id=msg['id']).execute() | 
|  | if process_message(msg, dry_run) and not dry_run: | 
|  | msg_service.trash(userId='me', id=msg['id']).execute() | 
|  | except apiclient.errors.HttpError as ex: | 
|  | logging.error('API Client HTTP error: %s', ex) |