Support nested views
Change-Id: I13532a16efc6e970ab5a7c021ec4d77be98d3de8
This commit is contained in:
parent
da1e227c8c
commit
587740ec75
7
doc/source/view_nested.rst
Normal file
7
doc/source/view_nested.rst
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
.. _view_nested:
|
||||||
|
|
||||||
|
Nested View
|
||||||
|
===========
|
||||||
|
|
||||||
|
.. automodule:: view_nested
|
||||||
|
:members:
|
93
jenkins_jobs/modules/view_nested.py
Normal file
93
jenkins_jobs/modules/view_nested.py
Normal file
@ -0,0 +1,93 @@
|
|||||||
|
# Copyright 2019 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.
|
||||||
|
|
||||||
|
"""
|
||||||
|
The view nested module handles creating Jenkins Nested views.
|
||||||
|
|
||||||
|
To create a nested view specify ``nested`` in the ``view-type`` attribute
|
||||||
|
to the :ref:`view_nested` definition.
|
||||||
|
|
||||||
|
:View Parameters:
|
||||||
|
* **name** (`str`): The name of the view.
|
||||||
|
* **view-type** (`str`): The type of view.
|
||||||
|
* **description** (`str`): A description of the view. (default '')
|
||||||
|
* **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)
|
||||||
|
* **views** (`list`): The views to nest.
|
||||||
|
* **default-view** (`str`): Name of the view to use as the default from the
|
||||||
|
nested ones. (the first one by default)
|
||||||
|
* **columns** (`list`): List of columns to be shown in view. (default empty
|
||||||
|
list)
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
.. literalinclude::
|
||||||
|
/../../tests/views/fixtures/view_nested.yaml
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
import xml.etree.ElementTree as XML
|
||||||
|
import jenkins_jobs.modules.base
|
||||||
|
import jenkins_jobs.modules.helpers as helpers
|
||||||
|
from jenkins_jobs.xml_config import XmlViewGenerator
|
||||||
|
|
||||||
|
COLUMN_DICT = {
|
||||||
|
"status": "hudson.views.StatusColumn",
|
||||||
|
"weather": "hudson.views.WeatherColumn",
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class Nested(jenkins_jobs.modules.base.Base):
|
||||||
|
def root_xml(self, data):
|
||||||
|
root = XML.Element("hudson.plugins.nested__view.NestedView")
|
||||||
|
|
||||||
|
mapping = [
|
||||||
|
("name", "name", None),
|
||||||
|
("description", "description", ""),
|
||||||
|
("filter-executors", "filterExecutors", False),
|
||||||
|
("filter-queue", "filterQueue", False),
|
||||||
|
]
|
||||||
|
helpers.convert_mapping_to_xml(root, data, mapping, fail_required=True)
|
||||||
|
|
||||||
|
XML.SubElement(root, "properties", {"class": "hudson.model.View$PropertyList"})
|
||||||
|
|
||||||
|
v_xml = XML.SubElement(root, "views")
|
||||||
|
views = data.get("views", [])
|
||||||
|
|
||||||
|
xml_view_generator = XmlViewGenerator(self.registry)
|
||||||
|
xml_views = xml_view_generator.generateXML(views)
|
||||||
|
|
||||||
|
for xml_job in xml_views:
|
||||||
|
v_xml.append(xml_job.xml)
|
||||||
|
|
||||||
|
d_xml = XML.SubElement(root, "defaultView")
|
||||||
|
d_xml.text = data.get("default-view", views[0]["name"])
|
||||||
|
|
||||||
|
c_xml = XML.SubElement(root, "columns")
|
||||||
|
# there is a columns element in a columns element
|
||||||
|
c_xml = XML.SubElement(c_xml, "columns")
|
||||||
|
columns = data.get("columns", [])
|
||||||
|
|
||||||
|
for column in columns:
|
||||||
|
if column in COLUMN_DICT:
|
||||||
|
XML.SubElement(c_xml, COLUMN_DICT[column])
|
||||||
|
else:
|
||||||
|
raise ValueError(
|
||||||
|
"Unsupported column %s is not one of: %s."
|
||||||
|
% (column, ", ".join(COLUMN_DICT.keys()))
|
||||||
|
)
|
||||||
|
|
||||||
|
return root
|
@ -64,6 +64,7 @@ jenkins_jobs.projects =
|
|||||||
jenkins_jobs.views =
|
jenkins_jobs.views =
|
||||||
all=jenkins_jobs.modules.view_all:All
|
all=jenkins_jobs.modules.view_all:All
|
||||||
list=jenkins_jobs.modules.view_list:List
|
list=jenkins_jobs.modules.view_list:List
|
||||||
|
nested=jenkins_jobs.modules.view_nested:Nested
|
||||||
pipeline=jenkins_jobs.modules.view_pipeline:Pipeline
|
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
|
||||||
|
@ -18,10 +18,12 @@
|
|||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
import doctest
|
import doctest
|
||||||
|
import configparser
|
||||||
import io
|
import io
|
||||||
import json
|
import json
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
|
import pkg_resources
|
||||||
import re
|
import re
|
||||||
import xml.etree.ElementTree as XML
|
import xml.etree.ElementTree as XML
|
||||||
|
|
||||||
@ -45,6 +47,7 @@ from jenkins_jobs.modules import project_multibranch
|
|||||||
from jenkins_jobs.modules import project_multijob
|
from jenkins_jobs.modules import project_multijob
|
||||||
from jenkins_jobs.modules import view_all
|
from jenkins_jobs.modules import view_all
|
||||||
from jenkins_jobs.modules import view_list
|
from jenkins_jobs.modules import view_list
|
||||||
|
from jenkins_jobs.modules import view_nested
|
||||||
from jenkins_jobs.modules import view_pipeline
|
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
|
||||||
@ -169,7 +172,8 @@ class BaseScenariosTestCase(testscenarios.TestWithScenarios, BaseTestCase):
|
|||||||
scenarios = []
|
scenarios = []
|
||||||
fixtures_path = None
|
fixtures_path = None
|
||||||
|
|
||||||
def test_yaml_snippet(self):
|
@mock.patch("pkg_resources.iter_entry_points")
|
||||||
|
def test_yaml_snippet(self, mock):
|
||||||
if not self.in_filename:
|
if not self.in_filename:
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -187,6 +191,24 @@ class BaseScenariosTestCase(testscenarios.TestWithScenarios, BaseTestCase):
|
|||||||
self.addDetail("plugins-info", text_content(str(plugins_info)))
|
self.addDetail("plugins-info", text_content(str(plugins_info)))
|
||||||
|
|
||||||
parser = YamlParser(jjb_config)
|
parser = YamlParser(jjb_config)
|
||||||
|
e = pkg_resources.EntryPoint.parse
|
||||||
|
d = pkg_resources.Distribution()
|
||||||
|
config = configparser.ConfigParser()
|
||||||
|
config.read(os.path.dirname(__file__) + "/../setup.cfg")
|
||||||
|
groups = {}
|
||||||
|
for key in config["entry_points"]:
|
||||||
|
groups[key] = list()
|
||||||
|
for line in config["entry_points"][key].split("\n"):
|
||||||
|
if "" == line.strip():
|
||||||
|
continue
|
||||||
|
groups[key].append(e(line, dist=d))
|
||||||
|
|
||||||
|
def mock_iter_entry_points(group, name=None):
|
||||||
|
return (
|
||||||
|
entry for entry in groups[group] if name is None or name == entry.name
|
||||||
|
)
|
||||||
|
|
||||||
|
mock.side_effect = mock_iter_entry_points
|
||||||
registry = ModuleRegistry(jjb_config, plugins_info)
|
registry = ModuleRegistry(jjb_config, plugins_info)
|
||||||
registry.set_parser_data(parser.data)
|
registry.set_parser_data(parser.data)
|
||||||
|
|
||||||
@ -213,11 +235,13 @@ class BaseScenariosTestCase(testscenarios.TestWithScenarios, BaseTestCase):
|
|||||||
|
|
||||||
if "view-type" in yaml_content:
|
if "view-type" in yaml_content:
|
||||||
if yaml_content["view-type"] == "all":
|
if yaml_content["view-type"] == "all":
|
||||||
project = view_all.All(None)
|
project = view_all.All(registry)
|
||||||
elif yaml_content["view-type"] == "list":
|
elif yaml_content["view-type"] == "list":
|
||||||
project = view_list.List(None)
|
project = view_list.List(registry)
|
||||||
|
elif yaml_content["view-type"] == "nested":
|
||||||
|
project = view_nested.Nested(registry)
|
||||||
elif yaml_content["view-type"] == "pipeline":
|
elif yaml_content["view-type"] == "pipeline":
|
||||||
project = view_pipeline.Pipeline(None)
|
project = view_pipeline.Pipeline(registry)
|
||||||
else:
|
else:
|
||||||
raise InvalidAttributeError("view-type", yaml_content["view-type"])
|
raise InvalidAttributeError("view-type", yaml_content["view-type"])
|
||||||
|
|
||||||
|
24
tests/views/fixtures/view_nested.xml
Normal file
24
tests/views/fixtures/view_nested.xml
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<hudson.plugins.nested__view.NestedView>
|
||||||
|
<name>NestedViewTest</name>
|
||||||
|
<description/>
|
||||||
|
<filterExecutors>false</filterExecutors>
|
||||||
|
<filterQueue>false</filterQueue>
|
||||||
|
<properties class="hudson.model.View$PropertyList"/>
|
||||||
|
<views>
|
||||||
|
<hudson.model.AllView>
|
||||||
|
<name>All</name>
|
||||||
|
<description/>
|
||||||
|
<filterExecutors>false</filterExecutors>
|
||||||
|
<filterQueue>false</filterQueue>
|
||||||
|
<properties class="hudson.model.View$PropertyList"/>
|
||||||
|
</hudson.model.AllView>
|
||||||
|
</views>
|
||||||
|
<defaultView>All</defaultView>
|
||||||
|
<columns>
|
||||||
|
<columns>
|
||||||
|
<hudson.views.StatusColumn/>
|
||||||
|
<hudson.views.WeatherColumn/>
|
||||||
|
</columns>
|
||||||
|
</columns>
|
||||||
|
</hudson.plugins.nested__view.NestedView>
|
8
tests/views/fixtures/view_nested.yaml
Normal file
8
tests/views/fixtures/view_nested.yaml
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
name: NestedViewTest
|
||||||
|
view-type: nested
|
||||||
|
views:
|
||||||
|
- name: All
|
||||||
|
view-type: all
|
||||||
|
columns:
|
||||||
|
- status
|
||||||
|
- weather
|
@ -15,6 +15,7 @@
|
|||||||
import os
|
import os
|
||||||
from jenkins_jobs.modules import view_all
|
from jenkins_jobs.modules import view_all
|
||||||
from jenkins_jobs.modules import view_list
|
from jenkins_jobs.modules import view_list
|
||||||
|
from jenkins_jobs.modules import view_nested
|
||||||
from jenkins_jobs.modules import view_pipeline
|
from jenkins_jobs.modules import view_pipeline
|
||||||
from tests import base
|
from tests import base
|
||||||
|
|
||||||
@ -31,6 +32,12 @@ class TestCaseModuleViewList(base.BaseScenariosTestCase):
|
|||||||
klass = view_list.List
|
klass = view_list.List
|
||||||
|
|
||||||
|
|
||||||
|
class TestCaseModuleViewNested(base.BaseScenariosTestCase):
|
||||||
|
fixtures_path = os.path.join(os.path.dirname(__file__), "fixtures")
|
||||||
|
scenarios = base.get_scenarios(fixtures_path)
|
||||||
|
klass = view_nested.Nested
|
||||||
|
|
||||||
|
|
||||||
class TestCaseModuleViewPipeline(base.BaseScenariosTestCase):
|
class TestCaseModuleViewPipeline(base.BaseScenariosTestCase):
|
||||||
fixtures_path = os.path.join(os.path.dirname(__file__), "fixtures")
|
fixtures_path = os.path.join(os.path.dirname(__file__), "fixtures")
|
||||||
scenarios = base.get_scenarios(fixtures_path)
|
scenarios = base.get_scenarios(fixtures_path)
|
||||||
|
Loading…
Reference in New Issue
Block a user