Merge "Add view management functionality"

This commit is contained in:
Jenkins 2016-11-10 16:35:23 +00:00 committed by Gerrit Code Review
commit e47f9629bc
26 changed files with 755 additions and 31 deletions

View File

@ -178,6 +178,20 @@ the Job Templates in the Job Group will be realized. For example:
Would cause the jobs `project-name-unit-tests` and `project-name-perf-tests` to be created Would cause the jobs `project-name-unit-tests` and `project-name-perf-tests` to be created
in Jenkins. in Jenkins.
.. _views:
Views
^^^^^
A view is a particular way of displaying a specific set of jobs. To
create a view, you must define a view in a YAML file and have a variable called view-type with a valid value. It looks like this::
- view:
name: view-name
view-type: list
Views are processed differently than Jobs and therefore will not work within a `Project`_ or a `Job Template`_.
.. _macro: .. _macro:
Macro Macro
@ -494,4 +508,3 @@ Generally the sequence is:
#. builders (maven, freestyle, matrix, etc..) #. builders (maven, freestyle, matrix, etc..)
#. postbuilders (maven only, configured like :ref:`builders`) #. postbuilders (maven only, configured like :ref:`builders`)
#. publishers/reporters/notifications #. publishers/reporters/notifications

View File

@ -161,16 +161,17 @@ When you're satisfied with the generated XML from the test, you can run::
jenkins-jobs update /path/to/defs jenkins-jobs update /path/to/defs
which will upload the job definitions to Jenkins if needed. Jenkins Job which will upload the job and view definitions to Jenkins if needed. Jenkins
Builder maintains, for each host, a cache [#f1]_ of previously configured jobs, Job Builder maintains, for each host, a cache [#f1]_ of previously configured
so that you can run that command as often as you like, and it will only jobs and views, so that you can run that command as often as you like, and it
update the jobs configurations in Jenkins if the defined definitions has will only update the jobs configurations in Jenkins if the defined definitions
changed since the last time it was run. Note: if you modify a job has changed since the last time it was run. Note: if you modify a job
directly in Jenkins, jenkins-jobs will not know about it and will not directly in Jenkins, jenkins-jobs will not know about it and will not
update it. update it.
To update a specific list of jobs, simply pass the job names as additional To update a specific list of jobs/views, simply pass the job/view names as
arguments after the job definition path. To update Foo1 and Foo2 run:: additional arguments after the job definition path. To update Foo1 and Foo2
run::
jenkins-jobs update /path/to/defs Foo1 Foo2 jenkins-jobs update /path/to/defs Foo1 Foo2
@ -248,19 +249,25 @@ are denoted by starting from the root, relative by containing
the path separator, and patterns by having neither. the path separator, and patterns by having neither.
Patterns use simple shell globing to match directories. Patterns use simple shell globing to match directories.
Deleting Jobs Deleting Jobs/Views
^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^^
Jenkins Job Builder supports deleting jobs from Jenkins. Jenkins Job Builder supports deleting jobs and views from Jenkins.
To delete a specific job:: To delete a specific job::
jenkins-jobs delete Foo1 jenkins-jobs delete Foo1
To delete a list of jobs, simply pass them as additional To delete a list of jobs or views, simply pass them as additional
arguments after the command:: arguments after the command::
jenkins-jobs delete Foo1 Foo2 jenkins-jobs delete Foo1 Foo2
To delete only views or only jobs, simply add the argument
--views-only or --jobs-only after the command::
jenkins-jobs delete --views-only Foo1
jenkins-jobs delete --jobs-only Foo1
The ``update`` command includes a ``delete-old`` option to remove obsolete The ``update`` command includes a ``delete-old`` option to remove obsolete
jobs:: jobs::
@ -270,21 +277,31 @@ Obsolete jobs are jobs once managed by JJB (as distinguished by a special
comment that JJB appends to their description), that were not generated in this comment that JJB appends to their description), that were not generated in this
JJB run. JJB run.
There is also a command to delete **all** jobs. There is also a command to delete **all** jobs and/or views.
**WARNING**: Use with caution:: **WARNING**: Use with caution.
To delete **all** jobs and views::
jenkins-jobs delete-all jenkins-jobs delete-all
TO delete **all** jobs::
jenkins-jobs delete-all --jobs-only
To delete **all** views::
jenkins-jobs delete-all --views-only
Globbed Parameters Globbed Parameters
^^^^^^^^^^^^^^^^^^ ^^^^^^^^^^^^^^^^^^
Jenkins job builder supports globbed parameters to identify jobs from a set of Jenkins job builder supports globbed parameters to identify jobs from a set of
definition files. This feature only supports JJB managed jobs. definition files. This feature only supports JJB managed jobs.
To update jobs that only have 'foo' in their name:: To update jobs/views that only have 'foo' in their name::
jenkins-jobs update ./myjobs \*foo\* jenkins-jobs update ./myjobs \*foo\*
To delete jobs that only have 'foo' in their name:: To delete jobs/views that only have 'foo' in their name::
jenkins-jobs delete --path ./myjobs \*foo\* jenkins-jobs delete --path ./myjobs \*foo\*

7
doc/source/view_list.rst Normal file
View File

@ -0,0 +1,7 @@
.. view_list:
List View
=========
.. automodule:: view_list
:members:

View File

@ -61,6 +61,8 @@ class JenkinsManager(object):
self._plugins_list = jjb_config.builder['plugins_info'] self._plugins_list = jjb_config.builder['plugins_info']
self._jobs = None self._jobs = None
self._job_list = None self._job_list = None
self._views = None
self._view_list = None
self._jjb_config = jjb_config self._jjb_config = jjb_config
@property @property
@ -274,3 +276,144 @@ class JenkinsManager(object):
def parallel_update_job(self, job): def parallel_update_job(self, job):
self.update_job(job.name, job.output().decode('utf-8')) self.update_job(job.name, job.output().decode('utf-8'))
return (job.name, job.md5()) return (job.name, job.md5())
################
# View related #
################
@property
def views(self):
if self._views is None:
# populate views
self._views = self.jenkins.get_views()
return self._views
@property
def view_list(self):
if self._view_list is None:
self._view_list = set(view['name'] for view in self.views)
return self._view_list
def get_views(self, cache=True):
if not cache:
self._views = None
self._view_list = None
return self.views
def is_view(self, view_name):
# first use cache
if view_name in self.view_list:
return True
# if not exists, use jenkins
return self.jenkins.view_exists(view_name)
def delete_view(self, view_name):
if self.is_view(view_name):
logger.info("Deleting jenkins view {}".format(view_name))
self.jenkins.delete_view(view_name)
def delete_views(self, views):
if views is not None:
logger.info("Removing jenkins view(s): %s" % ", ".join(views))
for view in views:
self.delete_view(view)
if self.cache.is_cached(view):
self.cache.set(view, '')
self.cache.save()
def delete_all_views(self):
views = self.get_views()
# Jenkins requires at least one view present. Don't remove the first
# view as it is likely the default view.
views.pop(0)
logger.info("Number of views to delete: %d", len(views))
for view in views:
self.delete_view(view['name'])
# Need to clear the JJB cache after deletion
self.cache.clear()
def update_view(self, view_name, xml):
if self.is_view(view_name):
logger.info("Reconfiguring jenkins view {0}".format(view_name))
self.jenkins.reconfig_view(view_name, xml)
else:
logger.info("Creating jenkins view {0}".format(view_name))
self.jenkins.create_view(view_name, xml)
def update_views(self, xml_views, output=None, n_workers=None):
orig = time.time()
logger.info("Number of views generated: %d", len(xml_views))
xml_views.sort(key=operator.attrgetter('name'))
if output:
# ensure only wrapped once
if hasattr(output, 'write'):
output = utils.wrap_stream(output)
for view in xml_views:
if hasattr(output, 'write'):
# `output` is a file-like object
logger.info("View name: %s", view.name)
logger.debug("Writing XML to '{0}'".format(output))
try:
output.write(view.output())
except IOError as exc:
if exc.errno == errno.EPIPE:
# EPIPE could happen if piping output to something
# that doesn't read the whole input (e.g.: the UNIX
# `head` command)
return
raise
continue
output_fn = os.path.join(output, view.name)
logger.debug("Writing XML to '{0}'".format(output_fn))
with io.open(output_fn, 'w', encoding='utf-8') as f:
f.write(view.output().decode('utf-8'))
return xml_views, len(xml_views)
# Filter out the views that did not change
logging.debug('Filtering %d views for changed views',
len(xml_views))
step = time.time()
views = [view for view in xml_views
if self.changed(view)]
logging.debug("Filtered for changed views in %ss",
(time.time() - step))
if not views:
return [], 0
# Update the views
logging.debug('Updating views')
step = time.time()
p_params = [{'view': view} for view in views]
results = self.parallel_update_view(
n_workers=n_workers,
concurrent=p_params)
logging.debug("Parsing results")
# generalize the result parsing, as a concurrent view always returns a
# list
if len(p_params) in (1, 0):
results = [results]
for result in results:
if isinstance(result, Exception):
raise result
else:
# update in-memory cache
v_name, v_md5 = result
self.cache.set(v_name, v_md5)
# write cache to disk
self.cache.save()
logging.debug("Updated %d views in %ss",
len(views),
time.time() - step)
logging.debug("Total run took %ss", (time.time() - orig))
return views, len(views)
@concurrent
def parallel_update_view(self, view):
self.update_view(view.name, view.output().decode('utf-8'))
return (view.name, view.md5())

View File

@ -15,6 +15,7 @@
from jenkins_jobs.builder import JenkinsManager from jenkins_jobs.builder import JenkinsManager
from jenkins_jobs.errors import JenkinsJobsException
from jenkins_jobs.parser import YamlParser from jenkins_jobs.parser import YamlParser
from jenkins_jobs.registry import ModuleRegistry from jenkins_jobs.registry import ModuleRegistry
import jenkins_jobs.cli.subcommand.base as base import jenkins_jobs.cli.subcommand.base as base
@ -36,10 +37,26 @@ class DeleteSubCommand(base.BaseSubCommand):
default=None, default=None,
help="colon-separated list of paths to YAML files " help="colon-separated list of paths to YAML files "
"or directories") "or directories")
delete.add_argument(
'-j', '--jobs-only',
action='store_true', dest='del_jobs',
default=False,
help='delete only jobs'
)
delete.add_argument(
'-v', '--views-only',
action='store_true', dest='del_views',
default=False,
help='delete only views'
)
def execute(self, options, jjb_config): def execute(self, options, jjb_config):
builder = JenkinsManager(jjb_config) builder = JenkinsManager(jjb_config)
if options.del_jobs and options.del_views:
raise JenkinsJobsException(
'"--views-only" and "--jobs-only" cannot be used together.')
fn = options.path fn = options.path
registry = ModuleRegistry(jjb_config, builder.plugins_list) registry = ModuleRegistry(jjb_config, builder.plugins_list)
parser = YamlParser(jjb_config) parser = YamlParser(jjb_config)
@ -48,7 +65,15 @@ class DeleteSubCommand(base.BaseSubCommand):
parser.load_files(fn) parser.load_files(fn)
parser.expandYaml(registry, options.name) parser.expandYaml(registry, options.name)
jobs = [j['name'] for j in parser.jobs] jobs = [j['name'] for j in parser.jobs]
views = [v['name'] for v in parser.views]
else: else:
jobs = options.name jobs = options.name
views = options.name
builder.delete_jobs(jobs) if options.del_jobs:
builder.delete_jobs(jobs)
elif options.del_views:
builder.delete_views(views)
else:
builder.delete_jobs(jobs)
builder.delete_views(views)

View File

@ -19,6 +19,7 @@ import sys
from jenkins_jobs import utils from jenkins_jobs import utils
from jenkins_jobs.builder import JenkinsManager from jenkins_jobs.builder import JenkinsManager
from jenkins_jobs.errors import JenkinsJobsException
import jenkins_jobs.cli.subcommand.base as base import jenkins_jobs.cli.subcommand.base as base
@ -35,14 +36,43 @@ class DeleteAllSubCommand(base.BaseSubCommand):
self.parse_option_recursive_exclude(delete_all) self.parse_option_recursive_exclude(delete_all)
delete_all.add_argument(
'-j', '--jobs-only',
action='store_true', dest='del_jobs',
default=False,
help='delete only jobs'
)
delete_all.add_argument(
'-v', '--views-only',
action='store_true', dest='del_views',
default=False,
help='delete only views'
)
def execute(self, options, jjb_config): def execute(self, options, jjb_config):
builder = JenkinsManager(jjb_config) builder = JenkinsManager(jjb_config)
reach = set()
if options.del_jobs and options.del_views:
raise JenkinsJobsException(
'"--views-only" and "--jobs-only" cannot be used together.')
elif options.del_jobs and not options.del_views:
reach.add('jobs')
elif options.del_views and not options.del_jobs:
reach.add('views')
else:
reach.update(('jobs', 'views'))
if not utils.confirm( if not utils.confirm(
'Sure you want to delete *ALL* jobs from Jenkins ' 'Sure you want to delete *ALL* {} from Jenkins '
'server?\n(including those not managed by Jenkins ' 'server?\n(including those not managed by Jenkins '
'Job Builder)'): 'Job Builder)'.format(" AND ".join(reach))):
sys.exit('Aborted') sys.exit('Aborted')
logger.info("Deleting all jobs") if options.del_jobs:
builder.delete_all_jobs() logger.info("Deleting all jobs")
builder.delete_all_jobs()
if options.del_views:
logger.info("Deleting all views")
builder.delete_all_views()

View File

@ -45,6 +45,8 @@ class TestSubCommand(update.UpdateSubCommand):
def execute(self, options, jjb_config): def execute(self, options, jjb_config):
builder, xml_jobs = self._generate_xmljobs(options, jjb_config) builder, xml_jobs, xml_views = self._generate_xmljobs(
options, jjb_config)
builder.update_jobs(xml_jobs, output=options.output_dir, n_workers=1) builder.update_jobs(xml_jobs, output=options.output_dir, n_workers=1)
builder.update_views(xml_views, output=options.output_dir, n_workers=1)

View File

@ -21,6 +21,7 @@ from jenkins_jobs.builder import JenkinsManager
from jenkins_jobs.parser import YamlParser from jenkins_jobs.parser import YamlParser
from jenkins_jobs.registry import ModuleRegistry from jenkins_jobs.registry import ModuleRegistry
from jenkins_jobs.xml_config import XmlJobGenerator from jenkins_jobs.xml_config import XmlJobGenerator
from jenkins_jobs.xml_config import XmlViewGenerator
from jenkins_jobs.errors import JenkinsJobsException from jenkins_jobs.errors import JenkinsJobsException
import jenkins_jobs.cli.subcommand.base as base import jenkins_jobs.cli.subcommand.base as base
@ -75,21 +76,24 @@ class UpdateSubCommand(base.BaseSubCommand):
# Generate XML # Generate XML
parser = YamlParser(jjb_config) parser = YamlParser(jjb_config)
registry = ModuleRegistry(jjb_config, builder.plugins_list) registry = ModuleRegistry(jjb_config, builder.plugins_list)
xml_generator = XmlJobGenerator(registry) xml_job_generator = XmlJobGenerator(registry)
xml_view_generator = XmlViewGenerator(registry)
parser.load_files(options.path) parser.load_files(options.path)
registry.set_parser_data(parser.data) registry.set_parser_data(parser.data)
job_data_list = parser.expandYaml(registry, options.names) job_data_list, view_data_list = parser.expandYaml(
registry, options.names)
xml_jobs = xml_generator.generateXML(job_data_list) xml_jobs = xml_job_generator.generateXML(job_data_list)
xml_views = xml_view_generator.generateXML(view_data_list)
jobs = parser.jobs jobs = parser.jobs
step = time.time() step = time.time()
logging.debug('%d XML files generated in %ss', logging.debug('%d XML files generated in %ss',
len(jobs), str(step - orig)) len(jobs), str(step - orig))
return builder, xml_jobs return builder, xml_jobs, xml_views
def execute(self, options, jjb_config): def execute(self, options, jjb_config):
@ -97,12 +101,17 @@ class UpdateSubCommand(base.BaseSubCommand):
raise JenkinsJobsException( raise JenkinsJobsException(
'Number of workers must be equal or greater than 0') 'Number of workers must be equal or greater than 0')
builder, xml_jobs = self._generate_xmljobs(options, jjb_config) builder, xml_jobs, xml_views = self._generate_xmljobs(
options, jjb_config)
jobs, num_updated_jobs = builder.update_jobs( jobs, num_updated_jobs = builder.update_jobs(
xml_jobs, n_workers=options.n_workers) xml_jobs, n_workers=options.n_workers)
logger.info("Number of jobs updated: %d", num_updated_jobs) logger.info("Number of jobs updated: %d", num_updated_jobs)
views, num_updated_views = builder.update_views(
xml_views, n_workers=options.n_workers)
logger.info("Number of views updated: %d", num_updated_views)
keep_jobs = [job.name for job in xml_jobs] keep_jobs = [job.name for job in xml_jobs]
if options.delete_old: if options.delete_old:
n = builder.delete_old_managed(keep=keep_jobs) n = builder.delete_old_managed(keep=keep_jobs)

View File

@ -0,0 +1,102 @@
# Copyright 2015 Openstack Foundation
# 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 xml.etree.ElementTree as XML
import jenkins_jobs.modules.base
"""
The view list module handles creating Jenkins List views.
To create a list view specify ``list`` in the ``view-type`` attribute
to the :ref:`View-list` definition.
:View Parameters:
* **name** (`str`): The name of the view.
* **view-type** (`str`): The type of view.
* **description** (`str`): A description of the view. (optional)
* **filter-executors** (`bool`): Show only executors that can
execute the included views. (default false)
* **filter-queue** (`bool`): Show only included jobs in builder
queue. (default false)
* **job-name** (`list`): List of jobs to be included.
* **columns** (`list`): List of columns to be shown in view.
* **regex** (`str`): . Regular expression for selecting jobs
(optional)
* **recurse** (`bool`): Recurse in subfolders.(default false)
* **status-filter** (`bool`): Filter job list by enabled/disabled
status. (optional)
"""
COLUMN_DICT = {
'status': 'hudson.views.StatusColumn',
'weather': 'hudson.views.WeatherColumn',
'job': 'hudson.views.JobColumn',
'last-success': 'hudson.views.LastSuccessColumn',
'last-failure': 'hudson.views.LastFailureColumn',
'last-duration': 'hudson.views.LastDurationColumn',
'build-button': 'hudson.views.BuildButtonColumn',
'last-stable': 'hudson.views.LastStableColumn',
}
class List(jenkins_jobs.modules.base.Base):
sequence = 0
def root_xml(self, data):
root = XML.Element('hudson.model.ListView')
XML.SubElement(root, 'name').text = data['name']
desc_text = data.get('description', None)
if desc_text is not None:
XML.SubElement(root, 'description').text = desc_text
filterExecutors = data.get('filter-executors', False)
FE_element = XML.SubElement(root, 'filterExecutors')
FE_element.text = 'true' if filterExecutors else 'false'
filterQueue = data.get('filter-queue', False)
FQ_element = XML.SubElement(root, 'filterQueue')
FQ_element.text = 'true' if filterQueue else 'false'
XML.SubElement(root, 'properties',
{'class': 'hudson.model.View$PropertyList'})
jn_xml = XML.SubElement(root, 'jobNames')
jobnames = data.get('job-name', None)
XML.SubElement(jn_xml, 'comparator', {'class':
'hudson.util.CaseInsensitiveComparator'})
if jobnames is not None:
for jobname in jobnames:
XML.SubElement(jn_xml, 'string').text = str(jobname)
XML.SubElement(root, 'jobFilters')
c_xml = XML.SubElement(root, 'columns')
columns = data.get('columns', [])
for column in columns:
if column in COLUMN_DICT:
XML.SubElement(c_xml, COLUMN_DICT[column])
regex = data.get('regex', None)
if regex is not None:
XML.SubElement(root, 'includeRegex').text = regex
recurse = data.get('recurse', False)
R_element = XML.SubElement(root, 'recurse')
R_element.text = 'true' if recurse else 'false'
statusfilter = data.get('status-filter', None)
if statusfilter is not None:
SF_element = XML.SubElement(root, 'statusFilter')
SF_element.text = 'true' if statusfilter else 'false'
return root

View File

@ -0,0 +1,145 @@
# Copyright 2015 Openstack Foundation
# 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 xml.etree.ElementTree as XML
import jenkins_jobs.modules.base
"""
The view pipeline module handles creating Jenkins Build Pipeline views.
To create a list view specify ``list`` in the ``view-type`` attribute
to the :ref:`View-pipeline` definition.
Requires the Jenkins
:jenkins-wiki:`Build Pipeline Plugin <build+pipeline+plugin>`.
:View Parameters:
* **name** (`str`): The name of the view.
* **view-type** (`str`): The type of view.
* **description** (`str`): A description of the view. (optional)
* **filter-executors** (`bool`): Show only executors that can
execute the included views. (default false)
* **filter-queue** (`bool`): Show only included jobs in builder
queue. (default false)
* **first-job** (`str`): Parent Job in the view.
* **no-of-displayed-builds** (`str`): Number of builds to display.
(default 1)
* **title** (`str`): Build view title. (optional)
* **linkStyle** (`str`): Console output link style. Can be
'Lightbox', 'New Window', or 'This Window'. (default Lightbox)
* **css-Url** (`str`): Url for Custom CSS files (optional)
* **latest-job-only** (`bool`) Trigger only latest job.
(default false)
* **manual-trigger** (`bool`) Always allow manual trigger.
(default false)
* **show-parameters** (`bool`) Show pipeline parameters.
(default false)
* **parameters-in-headers** (`bool`) Show pipeline parameters in
headers. (default false)
* **starts-with-parameters** (`bool`) Use Starts with parameters.
(default false)
* **refresh-frequency** (`str`) Frequency to refresh in seconds.
(default '3')
* **definition-header** (`bool`) Show pipeline definition header.
(default false)
Example:
.. literalinclude::
/../../tests/views/fixtures/pipeline_view001.yaml
Example:
.. literalinclude::
/../../tests/views/fixtures/pipeline_view002.yaml
"""
class Pipeline(jenkins_jobs.modules.base.Base):
sequence = 0
def root_xml(self, data):
linktypes = ['Lightbox', 'New Window']
root = XML.Element('au.com.centrumsystems.hudson.'
'plugin.buildpipeline.BuildPipelineView',
{'plugin': 'build-pipeline-plugin'})
XML.SubElement(root, 'name').text = data['name']
desc_text = data.get('description', None)
if desc_text is not None:
XML.SubElement(root, 'description').text = desc_text
filterExecutors = data.get('filter-executors', False)
FE_element = XML.SubElement(root, 'filterExecutors')
FE_element.text = 'true' if filterExecutors else 'false'
filterQueue = data.get('filter-queue', False)
FQ_element = XML.SubElement(root, 'filterQueue')
FQ_element.text = 'true' if filterQueue else 'false'
XML.SubElement(root, 'properties',
{'class': 'hudson.model.View$PropertyList'})
GBurl = ('au.com.centrumsystems.hudson.plugin.buildpipeline.'
'DownstreamProjectGridBuilder')
gridBuilder = XML.SubElement(root, 'gridBuilder', {'class': GBurl})
jobname = data.get('first-job', '')
XML.SubElement(gridBuilder, 'firstJob').text = jobname
builds = str(data.get('no-of-displayed-builds', 1))
XML.SubElement(root, 'noOfDisplayedBuilds').text = builds
title = data.get('title', None)
BVT_element = XML.SubElement(root, 'buildViewTitle')
if title is not None:
BVT_element.text = title
linkStyle = data.get('link-style', 'Lightbox')
LS_element = XML.SubElement(root, 'consoleOutputLinkStyle')
if linkStyle in linktypes:
LS_element.text = linkStyle
else:
LS_element.text = 'Lightbox'
cssUrl = data.get('css-Url', None)
CU_element = XML.SubElement(root, 'cssUrl')
if cssUrl is not None:
CU_element.text = cssUrl
latest_job_only = data.get('latest-job-only', False)
OLJ_element = XML.SubElement(root, 'triggerOnlyLatestJob')
OLJ_element.text = 'true' if latest_job_only else 'false'
manual_trigger = data.get('manual-trigger', False)
AMT_element = XML.SubElement(root, 'alwaysAllowManualTrigger')
AMT_element.text = 'true' if manual_trigger else 'false'
show_parameters = data.get('show-parameters', False)
PP_element = XML.SubElement(root, 'showPipelineParameters')
PP_element.text = 'true' if show_parameters else 'false'
parameters_in_headers = data.get('parameters-in-headers', False)
PIH_element = XML.SubElement(root, 'showPipelineParametersInHeaders')
PIH_element.text = 'true' if parameters_in_headers else 'false'
start_with_parameters = data.get('start-with-parameters', False)
SWP_element = XML.SubElement(root, 'startsWithParameters')
SWP_element.text = 'true' if start_with_parameters else 'false'
refresh_frequency = str(data.get('refresh-frequency', 3))
XML.SubElement(root, 'refreshFrequency').text = refresh_frequency
headers = data.get('definition-header', False)
DH_element = XML.SubElement(root, 'showPipelineDefinitionHeader')
DH_element.text = 'true' if headers else 'false'
return root

View File

@ -75,6 +75,7 @@ class YamlParser(object):
def __init__(self, jjb_config=None): def __init__(self, jjb_config=None):
self.data = {} self.data = {}
self.jobs = [] self.jobs = []
self.views = []
self.jjb_config = jjb_config self.jjb_config = jjb_config
self.keep_desc = jjb_config.yamlparser['keep_descriptions'] self.keep_desc = jjb_config.yamlparser['keep_descriptions']
@ -234,6 +235,12 @@ class YamlParser(object):
job = self._applyDefaults(job) job = self._applyDefaults(job)
self._formatDescription(job) self._formatDescription(job)
self.jobs.append(job) self.jobs.append(job)
for view in self.data.get('view', {}).values():
logger.debug("Expanding view '{0}'".format(view['name']))
self._formatDescription(view)
self.views.append(view)
for project in self.data.get('project', {}).values(): for project in self.data.get('project', {}).values():
logger.debug("Expanding project '{0}'".format(project['name'])) logger.debug("Expanding project '{0}'".format(project['name']))
# use a set to check for duplicate job references in projects # use a set to check for duplicate job references in projects
@ -308,7 +315,7 @@ class YamlParser(object):
"specified".format(job['name'])) "specified".format(job['name']))
self.jobs.remove(job) self.jobs.remove(job)
seen.add(job['name']) seen.add(job['name'])
return self.jobs return self.jobs, self.views
def _expandYamlForTemplateJob(self, project, template, jobs_glob=None): def _expandYamlForTemplateJob(self, project, template, jobs_glob=None):
dimensions = [] dimensions = []

View File

@ -96,3 +96,35 @@ class XmlJobGenerator(object):
for module in self.registry.modules: for module in self.registry.modules:
if hasattr(module, 'gen_xml'): if hasattr(module, 'gen_xml'):
module.gen_xml(xml, data) module.gen_xml(xml, data)
class XmlViewGenerator(object):
""" This class is responsible for generating Jenkins Configuration XML from
a compatible intermediate representation of Jenkins Views.
"""
def __init__(self, registry):
self.registry = registry
def generateXML(self, viewdict_list):
xml_views = []
for view in viewdict_list:
xml_views.append(self.__getXMLForView(view))
return xml_views
def __getXMLForView(self, data):
kind = data.get('view-type', 'list')
for ep in pkg_resources.iter_entry_points(
group='jenkins_jobs.views', name=kind):
Mod = ep.load()
mod = Mod(self.registry)
xml = mod.root_xml(data)
self.__gen_xml(xml, data)
view = XmlJob(xml, data['name'])
return view
def __gen_xml(self, xml, data):
for module in self.registry.modules:
if hasattr(module, 'gen_xml'):
module.gen_xml(xml, data)

View File

@ -48,6 +48,9 @@ jenkins_jobs.projects =
maven=jenkins_jobs.modules.project_maven:Maven maven=jenkins_jobs.modules.project_maven:Maven
multijob=jenkins_jobs.modules.project_multijob:MultiJob multijob=jenkins_jobs.modules.project_multijob:MultiJob
workflow=jenkins_jobs.modules.project_workflow:Workflow workflow=jenkins_jobs.modules.project_workflow:Workflow
jenkins_jobs.views =
list=jenkins_jobs.modules.view_list:List
pipeline=jenkins_jobs.modules.view_pipeline:Pipeline
jenkins_jobs.builders = jenkins_jobs.builders =
raw=jenkins_jobs.modules.general:raw raw=jenkins_jobs.modules.general:raw
jenkins_jobs.reporters = jenkins_jobs.reporters =

View File

@ -34,12 +34,15 @@ import testscenarios
from yaml import safe_dump from yaml import safe_dump
from jenkins_jobs.config import JJBConfig from jenkins_jobs.config import JJBConfig
from jenkins_jobs.errors import InvalidAttributeError
import jenkins_jobs.local_yaml as yaml import jenkins_jobs.local_yaml as yaml
from jenkins_jobs.modules import project_externaljob from jenkins_jobs.modules import project_externaljob
from jenkins_jobs.modules import project_flow from jenkins_jobs.modules import project_flow
from jenkins_jobs.modules import project_matrix from jenkins_jobs.modules import project_matrix
from jenkins_jobs.modules import project_maven from jenkins_jobs.modules import project_maven
from jenkins_jobs.modules import project_multijob from jenkins_jobs.modules import project_multijob
from jenkins_jobs.modules import view_list
from jenkins_jobs.modules import view_pipeline
from jenkins_jobs.parser import YamlParser from jenkins_jobs.parser import YamlParser
from jenkins_jobs.registry import ModuleRegistry from jenkins_jobs.registry import ModuleRegistry
from jenkins_jobs.xml_config import XmlJob from jenkins_jobs.xml_config import XmlJob
@ -175,6 +178,15 @@ class BaseScenariosTestCase(testscenarios.TestWithScenarios, BaseTestCase):
elif (yaml_content['project-type'] == "externaljob"): elif (yaml_content['project-type'] == "externaljob"):
project = project_externaljob.ExternalJob(registry) project = project_externaljob.ExternalJob(registry)
if 'view-type' in yaml_content:
if yaml_content['view-type'] == "list":
project = view_list.List(None)
elif yaml_content['view-type'] == "pipeline":
project = view_pipeline.Pipeline(None)
else:
raise InvalidAttributeError(
'view-type', yaml_content['view-type'])
if project: if project:
xml_project = project.root_xml(yaml_content) xml_project = project.root_xml(yaml_content)
else: else:
@ -206,7 +218,7 @@ class SingleJobTestCase(BaseScenariosTestCase):
registry = ModuleRegistry(config) registry = ModuleRegistry(config)
registry.set_parser_data(parser.data) registry.set_parser_data(parser.data)
job_data_list = parser.expandYaml(registry) job_data_list, view_data_list = parser.expandYaml(registry)
# Generate the XML tree # Generate the XML tree
xml_generator = XmlJobGenerator(registry) xml_generator = XmlJobGenerator(registry)

View File

@ -30,7 +30,9 @@ class DeleteTests(CmdTestsBase):
@mock.patch('jenkins_jobs.cli.subcommand.update.' @mock.patch('jenkins_jobs.cli.subcommand.update.'
'JenkinsManager.delete_jobs') 'JenkinsManager.delete_jobs')
def test_delete_single_job(self, delete_job_mock): @mock.patch('jenkins_jobs.cli.subcommand.update.'
'JenkinsManager.delete_views')
def test_delete_single_job(self, delete_job_mock, delete_view_mock):
""" """
Test handling the deletion of a single Jenkins job. Test handling the deletion of a single Jenkins job.
""" """
@ -40,7 +42,9 @@ class DeleteTests(CmdTestsBase):
@mock.patch('jenkins_jobs.cli.subcommand.update.' @mock.patch('jenkins_jobs.cli.subcommand.update.'
'JenkinsManager.delete_jobs') 'JenkinsManager.delete_jobs')
def test_delete_multiple_jobs(self, delete_job_mock): @mock.patch('jenkins_jobs.cli.subcommand.update.'
'JenkinsManager.delete_views')
def test_delete_multiple_jobs(self, delete_job_mock, delete_view_mock):
""" """
Test handling the deletion of multiple Jenkins jobs. Test handling the deletion of multiple Jenkins jobs.
""" """

View File

@ -123,6 +123,7 @@ class TestConfigs(CmdTestsBase):
args = ['--conf', self.default_config_file, 'update', path] args = ['--conf', self.default_config_file, 'update', path]
jenkins_mock.return_value.update_jobs.return_value = ([], 0) jenkins_mock.return_value.update_jobs.return_value = ([], 0)
jenkins_mock.return_value.update_views.return_value = ([], 0)
self.execute_jenkins_jobs_with_args(args) self.execute_jenkins_jobs_with_args(args)
# validate that the JJBConfig used to initialize builder.Jenkins # validate that the JJBConfig used to initialize builder.Jenkins
@ -146,6 +147,7 @@ class TestConfigs(CmdTestsBase):
args = ['--conf', config_file, 'update', path] args = ['--conf', config_file, 'update', path]
jenkins_mock.return_value.update_jobs.return_value = ([], 0) jenkins_mock.return_value.update_jobs.return_value = ([], 0)
jenkins_mock.return_value.update_views.return_value = ([], 0)
self.execute_jenkins_jobs_with_args(args) self.execute_jenkins_jobs_with_args(args)
# validate that the JJBConfig used to initialize builder.Jenkins # validate that the JJBConfig used to initialize builder.Jenkins

0
tests/views/__init__.py Normal file
View File

View File

@ -0,0 +1,27 @@
<?xml version="1.0" encoding="utf-8"?>
<hudson.model.ListView>
<name>list-view-name01</name>
<description>Sample description</description>
<filterExecutors>true</filterExecutors>
<filterQueue>true</filterQueue>
<properties class="hudson.model.View$PropertyList"/>
<jobNames>
<comparator class="hudson.util.CaseInsensitiveComparator"/>
<string>job-name-1</string>
<string>job-name-2</string>
<string>job-name-3</string>
</jobNames>
<jobFilters/>
<columns>
<hudson.views.StatusColumn/>
<hudson.views.WeatherColumn/>
<hudson.views.JobColumn/>
<hudson.views.LastSuccessColumn/>
<hudson.views.LastFailureColumn/>
<hudson.views.LastDurationColumn/>
<hudson.views.BuildButtonColumn/>
<hudson.views.LastStableColumn/>
</columns>
<recurse>true</recurse>
<statusFilter>false</statusFilter>
</hudson.model.ListView>

View File

@ -0,0 +1,20 @@
name: list-view-name01
view-type: list
description: 'Sample description'
filter-executors: true
filter-queue: true
job-name:
- job-name-1
- job-name-2
- job-name-3
columns:
- status
- weather
- job
- last-success
- last-failure
- last-duration
- build-button
- last-stable
recurse: true
status-filter: false

View File

@ -0,0 +1,21 @@
<?xml version="1.0" encoding="utf-8"?>
<hudson.model.ListView>
<name>regex-example</name>
<filterExecutors>false</filterExecutors>
<filterQueue>false</filterQueue>
<properties class="hudson.model.View$PropertyList"/>
<jobNames>
<comparator class="hudson.util.CaseInsensitiveComparator"/>
</jobNames>
<jobFilters/>
<columns>
<hudson.views.StatusColumn/>
<hudson.views.WeatherColumn/>
<hudson.views.JobColumn/>
<hudson.views.LastSuccessColumn/>
<hudson.views.LastFailureColumn/>
<hudson.views.LastDurationColumn/>
</columns>
<includeRegex>(?!test.*).*</includeRegex>
<recurse>false</recurse>
</hudson.model.ListView>

View File

@ -0,0 +1,10 @@
name: regex-example
view-type: list
columns:
- status
- weather
- job
- last-success
- last-failure
- last-duration
regex: (?!test.*).*

View File

@ -0,0 +1,22 @@
<?xml version="1.0" encoding="utf-8"?>
<au.com.centrumsystems.hudson.plugin.buildpipeline.BuildPipelineView plugin="build-pipeline-plugin">
<name>testBPview</name>
<description>This is a description</description>
<filterExecutors>false</filterExecutors>
<filterQueue>false</filterQueue>
<properties class="hudson.model.View$PropertyList"/>
<gridBuilder class="au.com.centrumsystems.hudson.plugin.buildpipeline.DownstreamProjectGridBuilder">
<firstJob>job-one</firstJob>
</gridBuilder>
<noOfDisplayedBuilds>5</noOfDisplayedBuilds>
<buildViewTitle>Title</buildViewTitle>
<consoleOutputLinkStyle>New Window</consoleOutputLinkStyle>
<cssUrl>fake.urlfor.css</cssUrl>
<triggerOnlyLatestJob>true</triggerOnlyLatestJob>
<alwaysAllowManualTrigger>true</alwaysAllowManualTrigger>
<showPipelineParameters>true</showPipelineParameters>
<showPipelineParametersInHeaders>true</showPipelineParametersInHeaders>
<startsWithParameters>true</startsWithParameters>
<refreshFrequency>3</refreshFrequency>
<showPipelineDefinitionHeader>true</showPipelineDefinitionHeader>
</au.com.centrumsystems.hudson.plugin.buildpipeline.BuildPipelineView>

View File

@ -0,0 +1,17 @@
name: testBPview
view-type: pipeline
description: 'This is a description'
filter-executors: false
filter-queue: false
first-job: job-one
no-of-displayed-builds: 5
title: Title
link-style: New Window
css-Url: fake.urlfor.css
latest-job-only: true
manual-trigger: true
show-parameters: true
parameters-in-headers: true
start-with-parameters: true
refresh-frequency: 3
definition-header: true

View File

@ -0,0 +1,21 @@
<?xml version="1.0" encoding="utf-8"?>
<au.com.centrumsystems.hudson.plugin.buildpipeline.BuildPipelineView plugin="build-pipeline-plugin">
<name>testBPview</name>
<filterExecutors>false</filterExecutors>
<filterQueue>false</filterQueue>
<properties class="hudson.model.View$PropertyList"/>
<gridBuilder class="au.com.centrumsystems.hudson.plugin.buildpipeline.DownstreamProjectGridBuilder">
<firstJob>job-one</firstJob>
</gridBuilder>
<noOfDisplayedBuilds>1</noOfDisplayedBuilds>
<buildViewTitle/>
<consoleOutputLinkStyle>Lightbox</consoleOutputLinkStyle>
<cssUrl/>
<triggerOnlyLatestJob>false</triggerOnlyLatestJob>
<alwaysAllowManualTrigger>false</alwaysAllowManualTrigger>
<showPipelineParameters>false</showPipelineParameters>
<showPipelineParametersInHeaders>false</showPipelineParametersInHeaders>
<startsWithParameters>false</startsWithParameters>
<refreshFrequency>3</refreshFrequency>
<showPipelineDefinitionHeader>false</showPipelineDefinitionHeader>
</au.com.centrumsystems.hudson.plugin.buildpipeline.BuildPipelineView>

View File

@ -0,0 +1,3 @@
name: testBPview
view-type: pipeline
first-job: job-one

30
tests/views/test_views.py Normal file
View File

@ -0,0 +1,30 @@
# Copyright 2015 Openstack Foundation
#
# 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 os
import os
from jenkins_jobs.modules import view_list
from jenkins_jobs.modules import view_pipeline
from tests import base
class TestCaseModuleViewList(base.BaseScenariosTestCase):
fixtures_path = os.path.join(os.path.dirname(__file__), 'fixtures')
scenarios = base.get_scenarios(fixtures_path)
klass = view_list.List
class TestCaseModuleViewPipeline(base.BaseScenariosTestCase):
fixtures_path = os.path.join(os.path.dirname(__file__), 'fixtures')
scenarios = base.get_scenarios(fixtures_path)
klass = view_pipeline.Pipeline