2017-03-14 18:26:34 +00:00
|
|
|
#!/usr/bin/env python2
|
|
|
|
|
|
|
|
import datetime
|
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import jinja2
|
|
|
|
import json
|
2017-03-23 02:56:28 +00:00
|
|
|
import tempfile
|
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
|
2017-03-16 19:26:22 +00:00
|
|
|
from google.cloud import logging
|
2017-03-14 18:26:34 +00:00
|
|
|
from google.cloud import storage
|
|
|
|
from jinja2 import Environment, FileSystemLoader
|
|
|
|
|
|
|
|
|
2017-03-23 02:56:28 +00:00
|
|
|
STATUS_BUCKET = 'oss-fuzz-build-logs'
|
2017-03-14 18:26:34 +00:00
|
|
|
LOGS_BUCKET = 'oss-fuzz-gcb-logs'
|
|
|
|
SCRIPT_DIR = os.path.dirname(os.path.abspath(__file__))
|
|
|
|
|
|
|
|
|
|
|
|
def usage():
|
|
|
|
sys.stderr.write(
|
|
|
|
"Usage: " + sys.argv[0] + " <projects_dir>\n")
|
|
|
|
exit(1)
|
|
|
|
|
|
|
|
|
|
|
|
def scan_project_names(projects_dir):
|
|
|
|
projects = []
|
|
|
|
for root, dirs, files in os.walk(projects_dir):
|
|
|
|
for f in files:
|
|
|
|
if f == "Dockerfile":
|
|
|
|
projects.append(os.path.basename(root))
|
|
|
|
return sorted(projects)
|
|
|
|
|
|
|
|
|
|
|
|
def upload_status(successes, failures, unstable):
|
|
|
|
"""Upload main status page."""
|
|
|
|
env = Environment(loader=FileSystemLoader(os.path.join(SCRIPT_DIR,
|
|
|
|
'templates')))
|
|
|
|
data = {
|
|
|
|
'projects': failures + successes + unstable,
|
|
|
|
'failures': failures,
|
|
|
|
'successes': successes,
|
|
|
|
'unstable': unstable,
|
|
|
|
'last_updated': datetime.datetime.utcnow().ctime()
|
|
|
|
}
|
|
|
|
|
|
|
|
storage_client = storage.Client()
|
2017-03-23 02:56:28 +00:00
|
|
|
bucket = storage_client.get_bucket(STATUS_BUCKET)
|
2017-03-14 18:26:34 +00:00
|
|
|
|
2017-03-14 21:05:25 +00:00
|
|
|
blob = bucket.blob('status.html')
|
|
|
|
blob.cache_control = 'no-cache'
|
|
|
|
blob.upload_from_string(
|
2017-03-14 18:26:34 +00:00
|
|
|
env.get_template('status_template.html').render(data),
|
|
|
|
content_type='text/html')
|
|
|
|
|
2017-03-14 21:05:25 +00:00
|
|
|
blob = bucket.blob('status.json')
|
|
|
|
blob.cache_control = 'no-cache'
|
|
|
|
blob.upload_from_string(
|
2017-03-14 18:26:34 +00:00
|
|
|
json.dumps(data),
|
|
|
|
content_type='text/html')
|
|
|
|
|
|
|
|
|
2017-03-16 19:26:22 +00:00
|
|
|
def is_build_successful(build):
|
|
|
|
if build['status'] == 'SUCCESS':
|
|
|
|
return True
|
|
|
|
|
|
|
|
build_id = build['id']
|
|
|
|
logging_client = logging.Client(project='clusterfuzz-external')
|
|
|
|
entries = logging_client.list_entries(
|
|
|
|
order_by=logging.DESCENDING,
|
|
|
|
page_size=1,
|
|
|
|
filter_=(
|
|
|
|
'resource.type="build" AND '
|
|
|
|
'resource.labels.build_id="{0}"'.format(build_id)))
|
|
|
|
|
|
|
|
entry = next(entries.pages)
|
|
|
|
entry = list(entry)[0]
|
|
|
|
return entry.payload == 'DONE'
|
|
|
|
|
|
|
|
|
2017-03-23 02:56:28 +00:00
|
|
|
def find_last_build(builds):
|
|
|
|
DELAY_MINUTES = 40
|
|
|
|
|
|
|
|
for build in builds:
|
|
|
|
finish_time = dateutil.parser.parse(build['finishTime'], ignoretz=True)
|
|
|
|
if (datetime.datetime.utcnow() - finish_time >=
|
|
|
|
datetime.timedelta(minutes=DELAY_MINUTES)):
|
|
|
|
storage_client = storage.Client()
|
|
|
|
|
|
|
|
status_bucket = storage_client.get_bucket(STATUS_BUCKET)
|
|
|
|
gcb_bucket = storage_client.get_bucket(LOGS_BUCKET)
|
|
|
|
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
|
|
|
|
|
|
|
|
|
2017-03-14 18:26:34 +00:00
|
|
|
def main():
|
|
|
|
if len(sys.argv) != 2:
|
|
|
|
usage()
|
|
|
|
|
|
|
|
projects_dir = sys.argv[1]
|
|
|
|
|
|
|
|
credentials = GoogleCredentials.get_application_default()
|
|
|
|
cloudbuild = gcb_build('cloudbuild', 'v1', credentials=credentials)
|
|
|
|
|
|
|
|
successes = []
|
|
|
|
failures = []
|
|
|
|
for project in scan_project_names(projects_dir):
|
|
|
|
print project
|
2017-03-16 19:26:22 +00:00
|
|
|
query_filter = ('(status="SUCCESS" OR status="FAILURE") AND ' +
|
2017-03-17 17:47:55 +00:00
|
|
|
'images="gcr.io/oss-fuzz/{0}"'.format(project))
|
2017-03-14 18:26:34 +00:00
|
|
|
response = cloudbuild.projects().builds().list(
|
|
|
|
projectId='clusterfuzz-external',
|
|
|
|
filter=query_filter).execute()
|
|
|
|
if not 'builds' in response:
|
|
|
|
continue
|
|
|
|
|
|
|
|
builds = response['builds']
|
2017-03-23 02:56:28 +00:00
|
|
|
last_build = find_last_build(builds)
|
|
|
|
if not last_build:
|
|
|
|
print >>sys.stderr, 'Failed to get build for', project
|
|
|
|
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-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-03-14 18:26:34 +00:00
|
|
|
|
|
|
|
upload_status(successes, failures, [])
|
|
|
|
|
2017-03-23 02:56:28 +00:00
|
|
|
|
2017-03-14 18:26:34 +00:00
|
|
|
if __name__ == "__main__":
|
|
|
|
main()
|
|
|
|
|