#!/usr/bin/env python2 # # 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 json import logging import os from apscheduler.schedulers.background import BackgroundScheduler from flask import Flask, request import requests import gerrit import tasks app = Flask(__name__) def gerrit_url(endpoint): gerrit_base_url = 'https://android-review.googlesource.com' return gerrit_base_url + endpoint @app.route('/', methods=['POST']) def handle_build_message(): result = json.loads(request.data) name = result['name'] number = result['build']['number'] status = result['build']['status'] go_url = 'http://go/bionicbb/' + result['build']['url'] full_url = result['build']['full_url'] params = result['build']['parameters'] change_id = params['CHANGE_ID'] ref = params['REF'] patch_set = ref.split('/')[-1] logging.debug('%s #%s %s: %s', name, number, status, full_url) # bionic-lint is always broken, so we don't want to reject changes for # those failures until we clean things up. if name == 'bionic-presubmit': message_lines = ['{} #{} checkbuild {}: {}'.format( name, number, status, go_url)] if status == 'FAILURE': message_lines += ['If you believe this Verified-1 was in error, ' '+1 the change and bionicbb will remove the -1 ' 'shortly.'] request_data = { 'message': '\n'.join(message_lines) } label = 'Verified' if status == 'FAILURE': request_data['labels'] = {label: -1} elif status == 'SUCCESS': request_data['labels'] = {label: +1} url = gerrit_url('/a/changes/{}/revisions/{}/review'.format(change_id, patch_set)) headers = {'Content-Type': 'application/json;charset=UTF-8'} logging.debug('POST %s: %s', url, request_data) requests.post(url, headers=headers, json=request_data) elif name == 'clean-bionic-presubmit': request_data = {'message': 'out/ directory removed'} url = gerrit_url('/a/changes/{}/revisions/{}/review'.format(change_id, patch_set)) headers = {'Content-Type': 'application/json;charset=UTF-8'} logging.debug('POST %s: %s', url, request_data) requests.post(url, headers=headers, json=request_data) elif name == 'bionic-lint': logging.warning('Result for bionic-lint ignored') else: logging.error('Unknown project: %s', name) return '' @app.route('/drop-rejection', methods=['POST']) def drop_rejection(): revision_info = json.loads(request.data) change_id = revision_info['changeid'] patch_set = revision_info['patchset'] bb_email = 'bionicbb@android.com' labels = gerrit.get_labels(change_id, patch_set) if bb_email in labels['Verified']: bb_review = labels['Verified'][bb_email] else: bb_review = 0 if bb_review >= 0: logging.info('No rejection to drop: %s %s', change_id, patch_set) return '' logging.info('Dropping rejection: %s %s', change_id, patch_set) request_data = {'labels': {'Verified': 0}} url = gerrit_url('/a/changes/{}/revisions/{}/review'.format(change_id, patch_set)) headers = {'Content-Type': 'application/json;charset=UTF-8'} logging.debug('POST %s: %s', url, request_data) requests.post(url, headers=headers, json=request_data) return '' if __name__ == "__main__": logging.basicConfig(level=logging.INFO) logger = logging.getLogger() fh = logging.FileHandler('bionicbb.log') fh.setLevel(logging.INFO) logger.addHandler(fh) # Prevent the job from being rescheduled by the reloader. if os.environ.get('WERKZEUG_RUN_MAIN') == 'true': scheduler = BackgroundScheduler() scheduler.start() scheduler.add_job(tasks.get_and_process_jobs, 'interval', minutes=5) app.run(host='0.0.0.0', debug=True)