2017-03-14 18:26:34 +00:00
|
|
|
#!/usr/bin/env python2
|
|
|
|
|
|
|
|
import datetime
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import json
|
2017-03-23 02:56:28 +00:00
|
|
|
import tempfile
|
2018-05-30 08:40:19 +00:00
|
|
|
import time
|
2017-03-14 18:26:34 +00:00
|
|
|
|
2017-03-23 02:56:28 +00:00
|
|
|
import dateutil.parser
|
2017-03-14 18:26:34 +00:00
|
|
|
from oauth2client.client import GoogleCredentials
|
|
|
|
from googleapiclient.discovery import build as gcb_build
|
|
|
|
from google.cloud import storage
|
|
|
|
|
2018-08-27 13:46:17 +00:00
|
|
|
import build_and_run_coverage
|
2018-08-24 02:14:09 +00:00
|
|
|
import build_project
|
|
|
|
|
2017-03-23 02:56:28 +00:00
|
|
|
STATUS_BUCKET = 'oss-fuzz-build-logs'
|
2017-03-14 18:26:34 +00:00
|
|
|
SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
|
2019-08-12 16:39:42 +00:00
|
|
|
BADGE_DIR = 'badges'
|
2018-05-30 08:40:19 +00:00
|
|
|
RETRY_COUNT = 3
|
|
|
|
RETRY_WAIT = 5
|
2019-04-02 04:00:55 +00:00
|
|
|
MAX_BUILD_RESULTS = 2000
|
|
|
|
BUILDS_PAGE_SIZE = 256
|
2019-08-23 20:43:10 +00:00
|
|
|
BADGE_IMAGE_TYPES = {'svg': 'image/svg+xml', 'png': 'image/png'}
|
|
|
|
|
|
|
|
_client = None
|
|
|
|
|
|
|
|
|
|
|
|
def _get_storage_client():
|
|
|
|
"""Return storage client."""
|
2019-08-23 20:44:21 +00:00
|
|
|
global _client
|
2019-08-23 20:43:10 +00:00
|
|
|
if not _client:
|
|
|
|
_client = storage.Client()
|
|
|
|
|
|
|
|
return _client
|
2017-03-14 18:26:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
def usage():
|
2018-07-27 03:32:09 +00:00
|
|
|
sys.stderr.write('Usage: ' + sys.argv[0] + ' <projects_dir>\n')
|
2017-03-14 18:26:34 +00:00
|
|
|
exit(1)
|
|
|
|
|
|
|
|
|
|
|
|
def scan_project_names(projects_dir):
|
|
|
|
projects = []
|
|
|
|
for root, dirs, files in os.walk(projects_dir):
|
|
|
|
for f in files:
|
2018-07-27 03:32:09 +00:00
|
|
|
if f == 'Dockerfile':
|
2017-03-14 18:26:34 +00:00
|
|
|
projects.append(os.path.basename(root))
|
|
|
|
return sorted(projects)
|
|
|
|
|
|
|
|
|
2018-08-27 13:46:17 +00:00
|
|
|
def upload_status(successes, failures, status_filename):
|
2017-03-14 18:26:34 +00:00
|
|
|
"""Upload main status page."""
|
|
|
|
data = {
|
2017-04-18 05:56:04 +00:00
|
|
|
'projects': failures + successes,
|
2017-03-14 18:26:34 +00:00
|
|
|
'failures': failures,
|
|
|
|
'successes': successes,
|
|
|
|
'last_updated': datetime.datetime.utcnow().ctime()
|
|
|
|
}
|
|
|
|
|
2019-08-23 20:43:10 +00:00
|
|
|
bucket = _get_storage_client().get_bucket(STATUS_BUCKET)
|
2018-08-27 13:46:17 +00:00
|
|
|
blob = bucket.blob(status_filename)
|
2017-03-14 21:05:25 +00:00
|
|
|
blob.cache_control = 'no-cache'
|
2018-07-27 03:32:09 +00:00
|
|
|
blob.upload_from_string(json.dumps(data), content_type='application/json')
|
2017-03-14 18:26:34 +00:00
|
|
|
|
|
|
|
|
2017-03-16 19:26:22 +00:00
|
|
|
def is_build_successful(build):
|
2017-10-30 00:15:51 +00:00
|
|
|
return build['status'] == 'SUCCESS'
|
2017-03-16 19:26:22 +00:00
|
|
|
|
|
|
|
|
2019-04-02 04:00:55 +00:00
|
|
|
def find_last_build(builds, project, build_tag_suffix):
|
2017-03-23 02:56:28 +00:00
|
|
|
DELAY_MINUTES = 40
|
2019-04-02 04:00:55 +00:00
|
|
|
tag = project + '-' + build_tag_suffix
|
|
|
|
|
|
|
|
builds = builds.get(tag)
|
|
|
|
if not builds:
|
2019-08-23 20:43:10 +00:00
|
|
|
print >> sys.stderr, 'Failed to find builds with tag', tag
|
2019-04-02 04:00:55 +00:00
|
|
|
return None
|
2017-03-23 02:56:28 +00:00
|
|
|
|
|
|
|
for build in builds:
|
2017-10-06 07:45:37 +00:00
|
|
|
if build['status'] == 'WORKING':
|
|
|
|
continue
|
|
|
|
|
2019-04-02 04:00:55 +00:00
|
|
|
if tag not in build['tags']:
|
|
|
|
continue
|
|
|
|
|
|
|
|
if not 'finishTime' in build:
|
|
|
|
continue
|
|
|
|
|
2017-03-23 02:56:28 +00:00
|
|
|
finish_time = dateutil.parser.parse(build['finishTime'], ignoretz=True)
|
|
|
|
if (datetime.datetime.utcnow() - finish_time >=
|
|
|
|
datetime.timedelta(minutes=DELAY_MINUTES)):
|
2019-08-23 20:43:10 +00:00
|
|
|
status_bucket = _get_storage_client().get_bucket(STATUS_BUCKET)
|
|
|
|
gcb_bucket = _get_storage_client().get_bucket(
|
|
|
|
build_project.GCB_LOGS_BUCKET)
|
2017-03-23 02:56:28 +00:00
|
|
|
log_name = 'log-{0}.txt'.format(build['id'])
|
|
|
|
log = gcb_bucket.blob(log_name)
|
|
|
|
dest_log = status_bucket.blob(log_name)
|
|
|
|
|
|
|
|
with tempfile.NamedTemporaryFile() as f:
|
|
|
|
log.download_to_filename(f.name)
|
|
|
|
dest_log.upload_from_filename(f.name, content_type='text/plain')
|
|
|
|
|
|
|
|
return build
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2018-05-30 08:40:19 +00:00
|
|
|
def execute_with_retries(request):
|
|
|
|
for i in xrange(RETRY_COUNT + 1):
|
|
|
|
try:
|
|
|
|
return request.execute()
|
|
|
|
except Exception as e:
|
|
|
|
print('request failed with {0}, retrying...'.format(str(e)))
|
|
|
|
if i < RETRY_COUNT:
|
|
|
|
time.sleep(RETRY_WAIT)
|
|
|
|
continue
|
2018-07-27 03:32:09 +00:00
|
|
|
|
2018-05-30 08:40:19 +00:00
|
|
|
raise
|
|
|
|
|
2018-07-27 03:32:09 +00:00
|
|
|
|
2019-04-02 04:00:55 +00:00
|
|
|
def get_builds(cloudbuild):
|
|
|
|
"""Get a batch of the latest builds (up to MAX_BUILD_RESULTS), grouped by
|
|
|
|
tag."""
|
|
|
|
ungrouped_builds = []
|
|
|
|
next_page_token = None
|
|
|
|
|
|
|
|
while True:
|
|
|
|
page_size = min(BUILDS_PAGE_SIZE, MAX_BUILD_RESULTS - len(ungrouped_builds))
|
|
|
|
response = execute_with_retries(cloudbuild.projects().builds().list(
|
|
|
|
projectId='oss-fuzz', pageSize=page_size, pageToken=next_page_token))
|
|
|
|
|
|
|
|
if not 'builds' in response:
|
2019-04-03 04:29:23 +00:00
|
|
|
print >> sys.stderr, 'Invalid response from builds list:', response
|
2019-04-02 04:00:55 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
ungrouped_builds.extend(response['builds'])
|
|
|
|
if len(ungrouped_builds) >= MAX_BUILD_RESULTS:
|
|
|
|
break
|
|
|
|
|
|
|
|
next_page_token = response.get('nextPageToken')
|
|
|
|
|
|
|
|
builds = {}
|
|
|
|
for build in ungrouped_builds:
|
|
|
|
for tag in build['tags']:
|
|
|
|
builds.setdefault(tag, []).append(build)
|
|
|
|
|
|
|
|
return builds
|
|
|
|
|
|
|
|
|
2019-08-23 20:43:10 +00:00
|
|
|
def update_build_status(builds, projects, build_tag_suffix, status_filename):
|
2017-03-14 18:26:34 +00:00
|
|
|
successes = []
|
|
|
|
failures = []
|
2019-04-02 04:00:55 +00:00
|
|
|
|
2018-08-27 13:46:17 +00:00
|
|
|
for project in projects:
|
2017-03-14 18:26:34 +00:00
|
|
|
print project
|
2018-08-24 02:14:09 +00:00
|
|
|
|
2019-04-02 04:00:55 +00:00
|
|
|
last_build = find_last_build(builds, project, build_tag_suffix)
|
2017-03-23 02:56:28 +00:00
|
|
|
if not last_build:
|
2018-07-27 03:32:09 +00:00
|
|
|
print >> sys.stderr, 'Failed to get build for', project
|
2017-03-23 02:56:28 +00:00
|
|
|
continue
|
|
|
|
|
2017-03-14 18:26:34 +00:00
|
|
|
print last_build['startTime'], last_build['status'], last_build['id']
|
2017-03-16 19:26:22 +00:00
|
|
|
if is_build_successful(last_build):
|
|
|
|
successes.append({
|
|
|
|
'name': project,
|
|
|
|
'build_id': last_build['id'],
|
2017-04-18 05:56:04 +00:00
|
|
|
'finish_time': last_build['finishTime'],
|
2017-04-18 15:52:37 +00:00
|
|
|
'success': True,
|
2017-03-16 19:26:22 +00:00
|
|
|
})
|
2017-03-14 18:26:34 +00:00
|
|
|
else:
|
2017-03-16 19:26:22 +00:00
|
|
|
failures.append({
|
|
|
|
'name': project,
|
|
|
|
'build_id': last_build['id'],
|
2017-04-18 05:56:04 +00:00
|
|
|
'finish_time': last_build['finishTime'],
|
2017-04-18 15:52:37 +00:00
|
|
|
'success': False,
|
2017-03-16 19:26:22 +00:00
|
|
|
})
|
2017-03-14 18:26:34 +00:00
|
|
|
|
2018-08-27 13:46:17 +00:00
|
|
|
upload_status(successes, failures, status_filename)
|
|
|
|
|
|
|
|
|
2019-08-12 16:25:22 +00:00
|
|
|
def update_build_badges(builds, projects, build_tag, coverage_tag):
|
|
|
|
for project in projects:
|
|
|
|
last_build = find_last_build(builds, project, build_tag)
|
|
|
|
last_coverage_build = find_last_build(builds, project, coverage_tag)
|
|
|
|
if not last_build or not last_coverage_build:
|
|
|
|
continue
|
|
|
|
|
|
|
|
badge = 'building'
|
|
|
|
if not is_build_successful(last_coverage_build):
|
|
|
|
badge = 'coverage_failing'
|
|
|
|
if not is_build_successful(last_build):
|
|
|
|
badge = 'failing'
|
|
|
|
|
|
|
|
print("[badge] {}: {}".format(project, badge))
|
|
|
|
|
2019-08-23 20:43:10 +00:00
|
|
|
for extension, mime_type in BADGE_IMAGE_TYPES.items():
|
2019-08-12 16:25:22 +00:00
|
|
|
badge_name = '{badge}.{extension}'.format(
|
|
|
|
badge=badge, extension=extension)
|
|
|
|
# Retrieve the image relative to this script's location
|
2019-08-23 20:43:10 +00:00
|
|
|
badge_file = os.path.join(SCRIPT_DIR, 'badge_images', badge_name)
|
2019-08-12 16:25:22 +00:00
|
|
|
|
|
|
|
# The uploaded blob name should look like `badges/project.png`
|
2019-08-12 16:39:42 +00:00
|
|
|
blob_name = '{badge_dir}/{project_name}.{extension}'.format(
|
2019-08-23 20:43:10 +00:00
|
|
|
badge_dir=BADGE_DIR, project_name=project, extension=extension)
|
2019-08-12 16:25:22 +00:00
|
|
|
|
2019-08-23 20:43:10 +00:00
|
|
|
status_bucket = _get_storage_client().get_bucket(STATUS_BUCKET)
|
2019-08-12 16:25:22 +00:00
|
|
|
badge_blob = status_bucket.blob(blob_name)
|
|
|
|
badge_blob.upload_from_filename(badge_file, content_type=mime_type)
|
|
|
|
|
|
|
|
|
2018-08-27 13:46:17 +00:00
|
|
|
def main():
|
|
|
|
if len(sys.argv) != 2:
|
|
|
|
usage()
|
|
|
|
|
|
|
|
projects_dir = sys.argv[1]
|
|
|
|
projects = scan_project_names(projects_dir)
|
|
|
|
|
|
|
|
credentials = GoogleCredentials.get_application_default()
|
|
|
|
cloudbuild = gcb_build('cloudbuild', 'v1', credentials=credentials)
|
|
|
|
|
2019-04-02 04:00:55 +00:00
|
|
|
builds = get_builds(cloudbuild)
|
2019-08-23 20:43:10 +00:00
|
|
|
update_build_status(
|
|
|
|
builds,
|
|
|
|
projects,
|
|
|
|
build_project.FUZZING_BUILD_TAG,
|
|
|
|
status_filename='status.json')
|
|
|
|
update_build_status(
|
|
|
|
builds,
|
|
|
|
projects,
|
|
|
|
build_and_run_coverage.COVERAGE_BUILD_TAG,
|
|
|
|
status_filename='status-coverage.json')
|
|
|
|
|
|
|
|
update_build_badges(
|
|
|
|
builds,
|
|
|
|
projects,
|
|
|
|
build_tag=build_project.FUZZING_BUILD_TAG,
|
|
|
|
coverage_tag=build_and_run_coverage.COVERAGE_BUILD_TAG)
|
2019-08-12 16:25:22 +00:00
|
|
|
|
2017-03-23 02:56:28 +00:00
|
|
|
|
2018-07-27 03:32:09 +00:00
|
|
|
if __name__ == '__main__':
|
2017-03-14 18:26:34 +00:00
|
|
|
main()
|