3e3996d32a
Add support for local tags which are application specific to allow including of other yaml files or code from scripts. Allows for code to be maintained and tested as seperate files, as well as reduces duplication of yaml code that cannot be macro'ed or easily templated by including it from a common file. Adds support for the following tags: 'include' - load file as yaml code 'include-raw' - load file as data (for scripts) 'include-raw-escaped' - load file as data with escaping braces '{}' as default for use with job-templates Use configuration file options to provide a search path for the files. - Test behaviour of yaml tags independent of any XML generation by comparing json result of yaml parsing to verify that certain tags do/don't recall the yaml.load() method. - Add examples for the include tags via addition tests for YamlParser class Inspired by http://stackoverflow.com/questions/528281/how-can-i-include-an-yaml-file-inside-another Change-Id: Ib90a07043112d4739d6529ceddbc9817668bcec0
186 lines
6.5 KiB
Python
186 lines
6.5 KiB
Python
#!/usr/bin/env python
|
|
#
|
|
# Joint copyright:
|
|
# - Copyright 2012,2013 Wikimedia Foundation
|
|
# - Copyright 2012,2013 Antoine "hashar" Musso
|
|
# - Copyright 2013 Arnaud Fabre
|
|
#
|
|
# 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 codecs
|
|
import logging
|
|
import os
|
|
import re
|
|
import doctest
|
|
import json
|
|
import operator
|
|
import testtools
|
|
import xml.etree.ElementTree as XML
|
|
from ConfigParser import ConfigParser
|
|
import jenkins_jobs.local_yaml as yaml
|
|
from jenkins_jobs.builder import XmlJob, YamlParser, ModuleRegistry
|
|
from jenkins_jobs.modules import (project_flow,
|
|
project_matrix,
|
|
project_maven,
|
|
project_multijob)
|
|
|
|
|
|
def get_scenarios(fixtures_path, in_ext='yaml', out_ext='xml'):
|
|
"""Returns a list of scenarios, each scenario being described
|
|
by two parameters (yaml and xml filenames by default).
|
|
- content of the fixture output file (aka expected)
|
|
"""
|
|
scenarios = []
|
|
files = os.listdir(fixtures_path)
|
|
input_files = [f for f in files if re.match(r'.*\.{0}$'.format(in_ext), f)]
|
|
|
|
for input_filename in input_files:
|
|
output_candidate = re.sub(r'\.{0}$'.format(in_ext),
|
|
'.{0}'.format(out_ext), input_filename)
|
|
# Make sure the input file has a output counterpart
|
|
if output_candidate not in files:
|
|
raise Exception(
|
|
"No {0} file named '{1}' to match {2} file '{3}'"
|
|
.format(out_ext.toupper(), output_candidate,
|
|
in_ext.toupper(), input_filename))
|
|
|
|
conf_candidate = re.sub(r'\.yaml$', '.conf', input_filename)
|
|
# If present, add the configuration file
|
|
if conf_candidate not in files:
|
|
conf_candidate = None
|
|
|
|
scenarios.append((input_filename, {
|
|
'in_filename': input_filename,
|
|
'out_filename': output_candidate,
|
|
'conf_filename': conf_candidate,
|
|
}))
|
|
|
|
return scenarios
|
|
|
|
|
|
class BaseTestCase(object):
|
|
scenarios = []
|
|
fixtures_path = None
|
|
|
|
# TestCase settings:
|
|
maxDiff = None # always dump text difference
|
|
longMessage = True # keep normal error message when providing our
|
|
|
|
logging.basicConfig()
|
|
|
|
def _read_content(self):
|
|
# Read XML content, assuming it is unicode encoded
|
|
xml_filepath = os.path.join(self.fixtures_path, self.out_filename)
|
|
xml_content = u"%s" % codecs.open(xml_filepath, 'r', 'utf-8').read()
|
|
|
|
yaml_filepath = os.path.join(self.fixtures_path, self.in_filename)
|
|
with file(yaml_filepath, 'r') as yaml_file:
|
|
yaml_content = yaml.load(yaml_file)
|
|
|
|
return (yaml_content, xml_content)
|
|
|
|
def test_yaml_snippet(self):
|
|
if not self.out_filename or not self.in_filename:
|
|
return
|
|
|
|
yaml_content, expected_xml = self._read_content()
|
|
project = None
|
|
if ('project-type' in yaml_content):
|
|
if (yaml_content['project-type'] == "maven"):
|
|
project = project_maven.Maven(None)
|
|
elif (yaml_content['project-type'] == "matrix"):
|
|
project = project_matrix.Matrix(None)
|
|
elif (yaml_content['project-type'] == "flow"):
|
|
project = project_flow.Flow(None)
|
|
elif (yaml_content['project-type'] == "multijob"):
|
|
project = project_multijob.MultiJob(None)
|
|
|
|
if project:
|
|
xml_project = project.root_xml(yaml_content)
|
|
else:
|
|
xml_project = XML.Element('project')
|
|
parser = YamlParser()
|
|
pub = self.klass(ModuleRegistry({}))
|
|
|
|
# Generate the XML tree directly with modules/general
|
|
pub.gen_xml(parser, xml_project, yaml_content)
|
|
|
|
# Prettify generated XML
|
|
pretty_xml = unicode(XmlJob(xml_project, 'fixturejob').output(),
|
|
'utf-8')
|
|
|
|
self.assertThat(
|
|
pretty_xml,
|
|
testtools.matchers.DocTestMatches(expected_xml,
|
|
doctest.ELLIPSIS |
|
|
doctest.NORMALIZE_WHITESPACE |
|
|
doctest.REPORT_NDIFF)
|
|
)
|
|
|
|
|
|
class SingleJobTestCase(BaseTestCase):
|
|
def test_yaml_snippet(self):
|
|
if not self.out_filename or not self.in_filename:
|
|
return
|
|
|
|
xml_filepath = os.path.join(self.fixtures_path, self.out_filename)
|
|
expected_xml = u"%s" % open(xml_filepath, 'r').read()
|
|
|
|
yaml_filepath = os.path.join(self.fixtures_path, self.in_filename)
|
|
|
|
if self.conf_filename:
|
|
config = ConfigParser()
|
|
conf_filepath = os.path.join(self.fixtures_path,
|
|
self.conf_filename)
|
|
config.readfp(open(conf_filepath))
|
|
else:
|
|
config = None
|
|
parser = YamlParser(config)
|
|
parser.parse(yaml_filepath)
|
|
|
|
# Generate the XML tree
|
|
parser.generateXML()
|
|
|
|
parser.jobs.sort(key=operator.attrgetter('name'))
|
|
|
|
# Prettify generated XML
|
|
pretty_xml = "\n".join(job.output() for job in parser.jobs)
|
|
|
|
self.assertThat(
|
|
pretty_xml,
|
|
testtools.matchers.DocTestMatches(expected_xml,
|
|
doctest.ELLIPSIS |
|
|
doctest.NORMALIZE_WHITESPACE |
|
|
doctest.REPORT_NDIFF)
|
|
)
|
|
|
|
|
|
class JsonTestCase(BaseTestCase):
|
|
|
|
def test_yaml_snippet(self):
|
|
if not self.out_filename or not self.in_filename:
|
|
return
|
|
|
|
yaml_content, expected_json = self._read_content()
|
|
|
|
pretty_json = json.dumps(yaml_content, indent=4,
|
|
separators=(',', ': '))
|
|
|
|
self.assertThat(
|
|
pretty_json,
|
|
testtools.matchers.DocTestMatches(expected_json,
|
|
doctest.ELLIPSIS |
|
|
doctest.NORMALIZE_WHITESPACE |
|
|
doctest.REPORT_NDIFF)
|
|
)
|