2020-06-08 16:40:44 -07:00
|
|
|
#!/usr/bin/env python3
|
2017-11-02 12:04:45 +01:00
|
|
|
|
|
|
|
# Copyright 2017 SUSE Linux GmbH
|
|
|
|
#
|
|
|
|
# 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 sys
|
|
|
|
|
|
|
|
import yaml
|
|
|
|
|
|
|
|
|
|
|
|
projects_yaml = 'zuul.d/projects.yaml'
|
|
|
|
projects = yaml.safe_load(open(projects_yaml))
|
|
|
|
|
|
|
|
|
|
|
|
def normalize(s):
|
|
|
|
"Normalize string for comparison."
|
|
|
|
return s.lower().replace("_", "-")
|
|
|
|
|
|
|
|
|
|
|
|
def check_projects_sorted():
|
|
|
|
"""Check that the projects are in alphabetical order per section."""
|
|
|
|
|
2017-11-03 19:04:01 +01:00
|
|
|
print("\nChecking project list for alphabetical order")
|
2017-11-02 12:04:45 +01:00
|
|
|
print("============================================")
|
|
|
|
|
|
|
|
errors = False
|
|
|
|
last = ""
|
|
|
|
for entry in projects:
|
|
|
|
current = entry['project']['name']
|
|
|
|
if (normalize(last) > normalize(current)):
|
2017-11-03 19:04:01 +01:00
|
|
|
print(" ERROR: Wrong alphabetical order: %(last)s, %(current)s" %
|
2017-11-02 12:04:45 +01:00
|
|
|
{"last": last, "current": current})
|
|
|
|
errors = True
|
|
|
|
last = current
|
2017-11-03 19:04:01 +01:00
|
|
|
|
|
|
|
if not errors:
|
|
|
|
print("... all fine.")
|
|
|
|
return errors
|
|
|
|
|
|
|
|
|
|
|
|
def check_release_jobs():
|
|
|
|
"""Minimal release job checks."""
|
|
|
|
|
|
|
|
release_templates = [
|
|
|
|
'release-openstack-server',
|
|
|
|
'publish-to-pypi',
|
|
|
|
'publish-to-pypi-neutron',
|
|
|
|
'publish-to-pypi-horizon',
|
|
|
|
'puppet-release-jobs',
|
|
|
|
'nodejs4-publish-to-npm',
|
|
|
|
'nodejs6-publish-to-npm',
|
|
|
|
'xstatic-publish-jobs'
|
|
|
|
]
|
|
|
|
|
|
|
|
errors = False
|
|
|
|
print("\nChecking release jobs")
|
|
|
|
print("======================")
|
|
|
|
for entry in projects:
|
|
|
|
project = entry['project']
|
|
|
|
name = project['name']
|
2018-01-25 10:17:33 -06:00
|
|
|
found = [tmpl for tmpl in project.get('templates', [])
|
2017-11-03 19:04:01 +01:00
|
|
|
if tmpl in release_templates]
|
|
|
|
if len(found) > 1:
|
|
|
|
errors = True
|
|
|
|
print(" ERROR: Found multiple release jobs for %s:" % name)
|
|
|
|
for x in found:
|
|
|
|
print(" %s" % x)
|
|
|
|
print(" Use only one of them.")
|
|
|
|
|
|
|
|
if not errors:
|
|
|
|
print("... all fine.")
|
2017-11-02 12:04:45 +01:00
|
|
|
return errors
|
|
|
|
|
|
|
|
|
2018-06-28 08:34:58 +02:00
|
|
|
def blacklist_jobs():
|
|
|
|
"""Check that certain jobs and templates are *not* used."""
|
|
|
|
|
|
|
|
# Currently only handles templates
|
|
|
|
blacklist_templates = [
|
|
|
|
'system-required'
|
|
|
|
]
|
|
|
|
|
|
|
|
errors = False
|
|
|
|
print("\nChecking for obsolete jobs and templates")
|
|
|
|
print("=========================================")
|
|
|
|
for entry in projects:
|
|
|
|
project = entry['project']
|
|
|
|
name = project['name']
|
|
|
|
# The openstack catch all contains system-required, so skip that one.
|
|
|
|
if name == "^openstack.*":
|
|
|
|
continue
|
2019-04-19 10:24:36 +02:00
|
|
|
if name.startswith("^(airship|"):
|
|
|
|
continue
|
2018-06-28 08:34:58 +02:00
|
|
|
found = [tmpl for tmpl in project.get('templates', [])
|
|
|
|
if tmpl in blacklist_templates]
|
|
|
|
if found:
|
|
|
|
errors = True
|
|
|
|
print(" ERROR: Found obsolete template for %s:" % name)
|
|
|
|
for x in found:
|
|
|
|
print(" %s" % x)
|
|
|
|
print(" Remove it.")
|
|
|
|
|
|
|
|
if not errors:
|
|
|
|
print("... all fine.")
|
|
|
|
return errors
|
|
|
|
|
|
|
|
|
2017-11-24 21:44:48 +01:00
|
|
|
def check_pipeline(project, job_pipeline, pipeline_name):
|
|
|
|
|
|
|
|
errors = False
|
|
|
|
|
|
|
|
for job in job_pipeline:
|
|
|
|
if isinstance(job, dict):
|
|
|
|
for name in job:
|
|
|
|
if ('voting' in job[name]
|
|
|
|
and (not job[name]['voting'])):
|
|
|
|
errors = True
|
|
|
|
print(" Found non-voting job in %s:" % pipeline_name)
|
|
|
|
print(" project: %s" % project['name'])
|
|
|
|
print(" job: %s" % name)
|
|
|
|
return errors
|
|
|
|
|
|
|
|
|
|
|
|
def check_pipelines(project, pipeline_name):
|
|
|
|
|
|
|
|
errors = False
|
|
|
|
|
|
|
|
if pipeline_name in project and 'jobs' in project[pipeline_name]:
|
|
|
|
errors = check_pipeline(project, project[pipeline_name]['jobs'],
|
|
|
|
pipeline_name)
|
|
|
|
return errors
|
|
|
|
|
|
|
|
|
|
|
|
def check_voting():
|
|
|
|
errors = False
|
|
|
|
print("\nChecking voting status of jobs")
|
|
|
|
print("==============================")
|
|
|
|
|
|
|
|
for entry in projects:
|
|
|
|
project = entry['project']
|
|
|
|
errors |= check_pipelines(project, 'gate')
|
2017-11-24 22:07:52 +01:00
|
|
|
errors |= check_pipelines(project, 'experimental')
|
2017-11-24 21:44:48 +01:00
|
|
|
errors |= check_pipelines(project, 'post')
|
|
|
|
errors |= check_pipelines(project, 'periodic')
|
2017-11-24 22:07:52 +01:00
|
|
|
errors |= check_pipelines(project, 'periodic-stable')
|
2017-11-24 21:44:48 +01:00
|
|
|
|
|
|
|
if errors:
|
|
|
|
print(" Note the following about non-voting jobs in pipelines:")
|
|
|
|
print(" * Never run non-voting jobs in gate pipeline, they just")
|
|
|
|
print(" waste resources, remove such jobs.")
|
|
|
|
print(" * Experimental, periodic, and post pipelines are always")
|
|
|
|
print(" non-voting. The 'voting: false' line is redundant, remove")
|
|
|
|
print(" it.")
|
|
|
|
else:
|
|
|
|
print("... all fine.")
|
|
|
|
return errors
|
|
|
|
|
|
|
|
|
2018-08-04 16:14:58 +02:00
|
|
|
def check_only_boilerplate():
|
|
|
|
"""Check for redundant boilerplate with not jobs."""
|
|
|
|
|
|
|
|
errors = False
|
|
|
|
print("\nChecking that every project has entries")
|
|
|
|
print("======================")
|
|
|
|
for entry in projects:
|
|
|
|
project = entry['project']
|
|
|
|
if len(project.keys()) <= 1:
|
|
|
|
name = project['name']
|
|
|
|
errors = True
|
|
|
|
print(" Found project %s with no jobs configured." % name)
|
|
|
|
|
|
|
|
if errors:
|
|
|
|
print("Errors found!\n")
|
|
|
|
print("Do not add projects with only names entry but no jobs,")
|
|
|
|
print("remove the entry completely - unless you forgot to add jobs.")
|
|
|
|
else:
|
|
|
|
print("... all fine.")
|
|
|
|
|
|
|
|
return errors
|
|
|
|
|
|
|
|
|
2017-11-02 12:04:45 +01:00
|
|
|
def check_all():
|
|
|
|
|
2018-06-21 16:58:19 -05:00
|
|
|
errors = check_projects_sorted()
|
2018-06-28 08:34:58 +02:00
|
|
|
errors = blacklist_jobs() or errors
|
2017-11-03 19:04:01 +01:00
|
|
|
errors = check_release_jobs() or errors
|
2017-11-24 21:44:48 +01:00
|
|
|
errors = check_voting() or errors
|
2018-08-04 16:14:58 +02:00
|
|
|
errors = check_only_boilerplate() or errors
|
2017-11-02 12:04:45 +01:00
|
|
|
|
|
|
|
if errors:
|
2017-11-03 19:04:01 +01:00
|
|
|
print("\nFound errors in zuul.d/projects.yaml!\n")
|
2017-11-02 12:04:45 +01:00
|
|
|
else:
|
2017-11-03 19:04:01 +01:00
|
|
|
print("\nNo errors found in zuul.d/projects.yaml!\n")
|
2017-11-02 12:04:45 +01:00
|
|
|
return errors
|
|
|
|
|
2017-11-03 19:04:01 +01:00
|
|
|
|
2017-11-02 12:04:45 +01:00
|
|
|
if __name__ == "__main__":
|
|
|
|
sys.exit(check_all())
|