998 lines
48 KiB
Python
998 lines
48 KiB
Python
|
#
|
||
|
# Copyright 2020 Red Hat, Inc. and/or its affiliates
|
||
|
# and other contributors as indicated by the @author tags.
|
||
|
#
|
||
|
# 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.
|
||
|
#
|
||
|
#
|
||
|
|
||
|
"""
|
||
|
Keycloak package for Python to assists with upgrading of Keycloak to
|
||
|
particular Wildfly tag / release.
|
||
|
|
||
|
Copyright 2020 Red Hat, Inc. and/or its affiliates
|
||
|
and other contributors as indicated by the @author tags.
|
||
|
|
||
|
To use, simply 'import wildfly.upgrade' and call the necessary routines.
|
||
|
"""
|
||
|
|
||
|
import colorlog, copy, itertools, logging, lxml.etree, os, os.path, re, sys
|
||
|
|
||
|
from packaging.version import parse as parseVersion
|
||
|
from shutil import copyfileobj
|
||
|
from subprocess import check_call, check_output
|
||
|
from tempfile import NamedTemporaryFile
|
||
|
from urllib.request import HTTPError, urlopen
|
||
|
|
||
|
__all__ = [
|
||
|
'getElementsByXPath',
|
||
|
'getKeycloakGitRepositoryRoot',
|
||
|
'getModuleLogger',
|
||
|
'getStepLogger',
|
||
|
'getTaskLogger',
|
||
|
'getPomDependencyByArtifactId',
|
||
|
'getPomProperty',
|
||
|
'getVersionOfPomDependency',
|
||
|
'getXmlRoot',
|
||
|
'isWellFormedWildflyTag',
|
||
|
'loadGavDictionaryFromGavFile',
|
||
|
'loadGavDictionaryFromXmlFile',
|
||
|
'saveUrlToNamedTemporaryFile'
|
||
|
'updateAdapterLicenseFile',
|
||
|
'updateMainKeycloakPomFile'
|
||
|
]
|
||
|
|
||
|
__author__ = "Jan Lieskovsky <jlieskov@redhat.com>"
|
||
|
__status__ = "Alpha"
|
||
|
__version__ = "0.0.1"
|
||
|
|
||
|
#
|
||
|
# Various data structures for the module
|
||
|
#
|
||
|
# Module loggers
|
||
|
_moduleLoggers = {}
|
||
|
# 'pom' namespace prefix definition for lxml
|
||
|
_pom_ns = "http://maven.apache.org/POM/4.0.0"
|
||
|
# Maven GAV (groupId:artifactId:version) related stuff
|
||
|
_gav_elements = ['groupId', 'artifactId', 'version']
|
||
|
_gav_delimiter = ':'
|
||
|
|
||
|
#
|
||
|
# Various base helper routines
|
||
|
#
|
||
|
|
||
|
def getKeycloakGitRepositoryRoot():
|
||
|
"""
|
||
|
Return the absolute path to the Keycloak git repository clone.
|
||
|
"""
|
||
|
return check_output(['git', 'rev-parse', '--show-toplevel']).decode('utf-8').rstrip()
|
||
|
|
||
|
def isWellFormedWildflyTag(tag):
|
||
|
"""
|
||
|
Well formed Wildfly & Wildfly Core tag seems to follow the patterns:
|
||
|
1) First a digit followed by a dot both of them exactly three times.
|
||
|
2) Followed:
|
||
|
a) Either by a "Final" suffix, e.g.: "20.0.0.Final",
|
||
|
b) Or by one of "Alpha", "Beta", "CR" suffices, followed by one digit
|
||
|
|
||
|
Verifies the tag provided as routine argument follows this schema.
|
||
|
|
||
|
Exits with error if not.
|
||
|
"""
|
||
|
if tag and not re.search(r'(\d\.){3}((Alpha|Beta|CR)\d|Final)', tag):
|
||
|
getModuleLogger().error("Invalid Wildfly tag '%s', exiting!" % tag)
|
||
|
sys.exit(1)
|
||
|
else:
|
||
|
return tag
|
||
|
|
||
|
def saveUrlToNamedTemporaryFile(baseUrl):
|
||
|
"""
|
||
|
Fetch URL specified as routine argument to named temporary file and
|
||
|
return the name of that file.
|
||
|
|
||
|
Otherwise, log an error and exit with failure if HTTP error occurred.
|
||
|
"""
|
||
|
try:
|
||
|
with urlopen(baseUrl) as response:
|
||
|
with NamedTemporaryFile(delete=False) as outfile:
|
||
|
copyfileobj(response, outfile)
|
||
|
return outfile.name
|
||
|
except HTTPError:
|
||
|
getModuleLogger().error("Failed to download the file from '%s'!. Double-check the URL and retry!" % baseUrl)
|
||
|
sys.exit(1)
|
||
|
|
||
|
return None
|
||
|
|
||
|
def _emptyNewLine():
|
||
|
"""
|
||
|
Print additional new line.
|
||
|
"""
|
||
|
print()
|
||
|
|
||
|
def _logErrorAndExitIf(errorMessage, condition):
|
||
|
"""
|
||
|
Log particular error message and exit with error if specified condition was
|
||
|
met.
|
||
|
"""
|
||
|
if condition:
|
||
|
_emptyNewLine()
|
||
|
getModuleLogger().error(errorMessage)
|
||
|
_emptyNewLine()
|
||
|
sys.exit(1)
|
||
|
|
||
|
#
|
||
|
# Logging facility for the module
|
||
|
#
|
||
|
|
||
|
def setupLogger(loggerName = 'upgrade-wildfly', loggerFormatter = '%(log_color)s[%(levelname)s] %(name)s: %(message)s'):
|
||
|
"""
|
||
|
Initialize logger with custom 'loggerName' and custom 'loggerFormatter'.
|
||
|
"""
|
||
|
stdOutLogHandler = logging.StreamHandler(sys.stdout)
|
||
|
loggerFormatter = colorlog.ColoredFormatter(loggerFormatter)
|
||
|
stdOutLogHandler.setFormatter(loggerFormatter)
|
||
|
logger = logging.getLogger(loggerName)
|
||
|
logger.addHandler(stdOutLogHandler)
|
||
|
logger.setLevel(logging.INFO)
|
||
|
|
||
|
return logger
|
||
|
|
||
|
def getLogger(loggerName = 'Upgrade Wildfly for Keycloak', loggerFormatter = '%(log_color)s[%(levelname)s] [%(name)s]: %(message)s'):
|
||
|
"""
|
||
|
Return instance of a logger with custom 'loggerName' and custom
|
||
|
'loggerFormatter' or setup such a logger if it doesn't exist yet.
|
||
|
"""
|
||
|
global _moduleLoggers
|
||
|
if not loggerName in _moduleLoggers:
|
||
|
_moduleLoggers[loggerName] = setupLogger(loggerName, loggerFormatter)
|
||
|
|
||
|
return _moduleLoggers[loggerName]
|
||
|
|
||
|
def getModuleLogger():
|
||
|
"""
|
||
|
Return global logger for the module.
|
||
|
"""
|
||
|
return getLogger()
|
||
|
|
||
|
def getTaskLogger(taskLoggerName):
|
||
|
"""
|
||
|
Return custom logger handling (sub)tasks.
|
||
|
"""
|
||
|
taskLogFormatter = '\n%(log_color)s[%(levelname)s] [%(name)s] Performing Task:\n\n\t%(message)s\n'
|
||
|
return getLogger(loggerName = taskLoggerName, loggerFormatter = taskLogFormatter)
|
||
|
|
||
|
def getStepLogger():
|
||
|
"""
|
||
|
Return custom logger handling steps within tasks.
|
||
|
"""
|
||
|
stepLoggerName = 'step'
|
||
|
stepLoggerFormatter = '\t%(log_color)s[%(levelname)s]: %(message)s'
|
||
|
|
||
|
return getLogger(stepLoggerName, stepLoggerFormatter)
|
||
|
|
||
|
#
|
||
|
# Various XML search related helper routines
|
||
|
#
|
||
|
|
||
|
def getElementsByXPath(xmlTree, xPath, nameSpace = { "pom" : "%s" % _pom_ns }):
|
||
|
"""
|
||
|
Given the XML tree return the list of elements matching the 'xPath' from
|
||
|
the XML 'nameSpace'. 'nameSpace' is optional argument. If not specified
|
||
|
defaults to the POM XML namespace.
|
||
|
|
||
|
Returns empty list if no such element specified by 'xPath' is found.
|
||
|
"""
|
||
|
return xmlTree.xpath(xPath, namespaces = nameSpace)
|
||
|
|
||
|
def getPomDependencyByArtifactId(xmlTree, artifactIdText):
|
||
|
"""
|
||
|
Given the XML tree return list of POM dependency elements matching
|
||
|
'artifactIdText' in the text of the element.
|
||
|
|
||
|
Returns empty list if no such element with 'artifactIdText' is found.
|
||
|
"""
|
||
|
return xmlTree.xpath('/pom:project/pom:dependencyManagement/pom:dependencies/pom:dependency/pom:artifactId[text()="%s"]' % artifactIdText, namespaces = { "pom" : "%s" % _pom_ns })
|
||
|
|
||
|
def getPomProperty(xmlTree, propertyText):
|
||
|
"""
|
||
|
Given the XML tree return list of POM property elements matching
|
||
|
'propertyText' in the text of the element.
|
||
|
|
||
|
Returns empty list if no such element with 'propertyText' is found.
|
||
|
"""
|
||
|
return xmlTree.xpath('/pom:project/pom:properties/pom:%s' % propertyText, namespaces = { "pom" : "%s" % _pom_ns })
|
||
|
|
||
|
def getVersionOfPomDependency(xmlElem, groupIdText, artifactIdText):
|
||
|
"""
|
||
|
Given the list of XML POM dependency elements, return the value of
|
||
|
'<version>' subelement if 'groupIdText' and 'artifactIdText' match the
|
||
|
value of groupId and artifactId subelements in the dependency.
|
||
|
|
||
|
Otherwise, return None.
|
||
|
"""
|
||
|
version = None
|
||
|
for entry in xmlElem:
|
||
|
dependencyElem = entry.getparent()
|
||
|
for subelem in list(dependencyElem):
|
||
|
if subelem.tag == '{%s}groupId' % _pom_ns and subelem.text != groupIdText:
|
||
|
break
|
||
|
if subelem.tag == '{%s}artifactId' % _pom_ns and subelem.text != artifactIdText:
|
||
|
break
|
||
|
if subelem.tag == '{%s}version' % _pom_ns:
|
||
|
version = subelem.text
|
||
|
break
|
||
|
|
||
|
return version
|
||
|
|
||
|
def getXmlRoot(filename):
|
||
|
"""
|
||
|
Return root element of the XML tree by parsing the content of 'filename'.
|
||
|
|
||
|
Exit with error in the case of a failure.
|
||
|
"""
|
||
|
try:
|
||
|
xmlRoot = lxml.etree.parse(filename).getroot()
|
||
|
return xmlRoot
|
||
|
except lxml.etree.XMLSyntaxError:
|
||
|
getXmlRootFailureMessage = (
|
||
|
"Failed to get the root element of the XML tree from '%s' file! "
|
||
|
"Ensure the file is not opened in another process, and retry!" %
|
||
|
filename
|
||
|
)
|
||
|
getModuleLogger().error(getXmlRootFailureMessage)
|
||
|
sys.exit(1)
|
||
|
|
||
|
#
|
||
|
# Common helper routines utilized by various tasks
|
||
|
# performed within a Wildfly upgrade
|
||
|
#
|
||
|
|
||
|
def getProductNamesForKeycloakPomProfile(profile = 'community'):
|
||
|
"""
|
||
|
Return values of <product.name> and <product.name.full> elements
|
||
|
of the specified Keycloak main pom.xml 'profile'
|
||
|
"""
|
||
|
(productName, productNameFull) = (None, None)
|
||
|
|
||
|
_logErrorAndExitIf(
|
||
|
"Invalid profile name '%s'! It can be only one of 'community' or 'product'!" % profile,
|
||
|
profile not in ['community', 'product']
|
||
|
)
|
||
|
|
||
|
# Absolute path to main Keycloak pom.xml within the repo
|
||
|
mainKeycloakPomPath = getKeycloakGitRepositoryRoot() + "/pom.xml"
|
||
|
keycloakPomXmlTreeRoot = getXmlRoot(mainKeycloakPomPath)
|
||
|
pomProfileIdElem = getElementsByXPath(keycloakPomXmlTreeRoot, '/pom:project/pom:profiles/pom:profile/pom:id[text()="%s"]' % profile)
|
||
|
_logErrorAndExitIf(
|
||
|
"Can't locate the '%s' profile in main Keycloak pom.xml file!" % profile,
|
||
|
len(pomProfileIdElem) != 1
|
||
|
)
|
||
|
|
||
|
pomProfileElem = pomProfileIdElem[0].getparent()
|
||
|
pomProductNameElem = getElementsByXPath(pomProfileElem, './pom:properties/pom:product.name')
|
||
|
_logErrorAndExitIf(
|
||
|
"Can't determine product name from '%s' profile of main Keycloak pom.xml file!" % profile,
|
||
|
len(pomProductNameElem) != 1
|
||
|
)
|
||
|
productName = pomProductNameElem[0].text
|
||
|
pomProductNameFullElem = getElementsByXPath(pomProfileElem, './pom:properties/pom:product.name')
|
||
|
_logErrorAndExitIf(
|
||
|
"Can't determine the full product name from '%s' profile of main Keycloak pom.xml file!" % profile,
|
||
|
len(pomProductNameFullElem) != 1
|
||
|
)
|
||
|
productNameFull = pomProductNameFullElem[0].text
|
||
|
|
||
|
return (productName, productNameFull)
|
||
|
|
||
|
def getNumericArtifactVersion(gavDictionary, gavDictionaryKey):
|
||
|
"""
|
||
|
Extract the numeric version of the 'gavDictionaryKey' GA artifact
|
||
|
from 'gavDictionary'.
|
||
|
|
||
|
1) Return dictionary value of 'gavDictionaryKey' directly
|
||
|
if it's type is not a dictionary again.
|
||
|
|
||
|
2) If the 'gavDictionaryKey' value is a child dictionary
|
||
|
containing exactly one key, namely the name of the POM
|
||
|
<property> to which the numeric version corresponds
|
||
|
to, return the numeric artifact version from the
|
||
|
subdictionary value.
|
||
|
"""
|
||
|
gavDictionaryValue = gavDictionary[gavDictionaryKey]
|
||
|
if not isinstance(gavDictionaryValue, dict):
|
||
|
# First check if obtained artifact version is really numeric
|
||
|
_logErrorAndExitIf(
|
||
|
"Extracted '%s' artifact version isn't numeric: '%s'!" % (gavDictionaryKey, gavDictionaryValue),
|
||
|
not re.match(r'\d.*', gavDictionaryValue)
|
||
|
)
|
||
|
return gavDictionaryValue
|
||
|
|
||
|
else:
|
||
|
subKey = gavDictionaryValue.keys()
|
||
|
# Python starting from 3.3.1 returns dict_keys instead of a list when
|
||
|
# calling dictionary items(). Convert dict_keys back to list if needed
|
||
|
if not isinstance(subKey, list):
|
||
|
subKey = list(subKey)
|
||
|
# Sanity check if there's just one candidate numeric version for
|
||
|
# the artifact. This shouldn't ever happen, but better to check
|
||
|
_logErrorAndExitIf(
|
||
|
"Artifact '%s' can't have more than just one versions!" % gavDictionaryKey,
|
||
|
len(subKey) != 1
|
||
|
)
|
||
|
# Fetch the numeric artifact version from the subdictionary value
|
||
|
gavDictionaryValue = gavDictionary[gavDictionaryKey][subKey[0]]
|
||
|
# Finally check if obtained artifact version is really numeric
|
||
|
_logErrorAndExitIf(
|
||
|
"Extracted '%s' artifact version isn't numeric: '%s'!" % (gavDictionaryKey, gavDictionaryValue),
|
||
|
not re.match(r'\d.*', gavDictionaryValue)
|
||
|
)
|
||
|
return gavDictionaryValue
|
||
|
|
||
|
def loadGavDictionaryFromGavFile(gavFile):
|
||
|
"""
|
||
|
Load the content of 'gavFile' into Maven GAV Python dictionary, where
|
||
|
dictionary key is reppresented by 'groupId:artifactId' part of the GAV
|
||
|
entry, and value is represented by the 'version' field of the GAV entry.
|
||
|
"""
|
||
|
gavDictionary = {}
|
||
|
with open(gavFile) as inputFile:
|
||
|
for line in inputFile:
|
||
|
try:
|
||
|
groupId, artifactId, version = line.rstrip().split(_gav_delimiter, 3)
|
||
|
gavDictionaryKey = groupId + _gav_delimiter + artifactId
|
||
|
gavDictionaryValue = version
|
||
|
# Exit with error if obtained artifact version doesn't start
|
||
|
# with a number
|
||
|
_logErrorAndExitIf(
|
||
|
"Extracted '%s' artifact version isn't numeric: '%s'!" % (gavDictionaryKey, gavDictionaryValue),
|
||
|
not re.match(r'\d.*', gavDictionaryValue)
|
||
|
)
|
||
|
gavDictionary[gavDictionaryKey] = gavDictionaryValue
|
||
|
except ValueError:
|
||
|
# Ignore malformed GAV entries containing more than three
|
||
|
# fields separated by the ':' character
|
||
|
continue
|
||
|
|
||
|
return gavDictionary
|
||
|
|
||
|
def loadGavDictionaryFromXmlFile(xmlFile, xPathPrefix = '/pom:project/pom:dependencyManagement/pom:dependencies/pom:dependency/pom:', nameSpace = { "pom" : "%s" % _pom_ns }):
|
||
|
"""
|
||
|
Convert XML dependencies from 'xmlFile' into Maven GAV
|
||
|
(groupId:artifactId:version) Python dictionary, where
|
||
|
dictionary key is represented by 'groupId:artifactId'
|
||
|
part of the GAV entry, and value is:
|
||
|
|
||
|
* Either 'version' field of the GAV entry directly,
|
||
|
if the version is numeric,
|
||
|
|
||
|
* Or another child dictionary in the case the 'version' field
|
||
|
of the GAV entry represents a property within the
|
||
|
XML file. In this case, the key of the child dictionary
|
||
|
item is the name of such a XML <property> element.
|
||
|
The value of the child dictionary item is the
|
||
|
value of the <property> itself.
|
||
|
|
||
|
Returns GAV dictionary corresponding to 'xmlFile'
|
||
|
or exits with error in case of a failure
|
||
|
"""
|
||
|
xmlRoot = getXmlRoot(xmlFile)
|
||
|
# Construct the final union xPath query returning all three
|
||
|
# (GAV) subelements of a particular dependency element
|
||
|
gavXPathQuery = '|'.join(map(lambda x: xPathPrefix + x, _gav_elements))
|
||
|
xmlDependencyElements = getElementsByXPath(xmlRoot, gavXPathQuery, nameSpace)
|
||
|
_logErrorAndExitIf(
|
||
|
"Failed to load dependencies from XML file '%s'!" % xmlFile,
|
||
|
len(xmlDependencyElements) == 0
|
||
|
)
|
||
|
gavDictionary = {}
|
||
|
# Divide original list into sublists by three elements -- one sublist per GAV entry
|
||
|
for gavEntry in [xmlDependencyElements[i:i + 3] for i in range(0, len(xmlDependencyElements), 3)]:
|
||
|
(groupIdElem, artifactIdElem, versionElem) = (gavEntry[0], gavEntry[1], gavEntry[2])
|
||
|
_logErrorAndExitIf(
|
||
|
"Failed to load '%s' dependency from XML file!" % gavEntry,
|
||
|
groupIdElem is None or artifactIdElem is None or versionElem is None
|
||
|
)
|
||
|
gavDictKey = groupIdElem.text + _gav_delimiter + artifactIdElem.text
|
||
|
gavDictValue = versionElem.text
|
||
|
if re.match(r'\d.*', gavDictValue):
|
||
|
# Store the numeric artifact version into GAV dictionary
|
||
|
gavDictionary[gavDictKey] = gavDictValue
|
||
|
else:
|
||
|
childDictKey = gavDictValue
|
||
|
while not re.match(r'\d.*', gavDictValue):
|
||
|
gavDictValue = re.sub(r'^\${', '', gavDictValue)
|
||
|
gavDictValue = re.sub(r'}$', '', gavDictValue)
|
||
|
propertyElem = getPomProperty(xmlRoot, gavDictValue)
|
||
|
# Handle corner case when artifact version isn't value of some POM <property> element,
|
||
|
# but rather value of some xPath within the XML file. Like for example the case of
|
||
|
# 'project.version' value. Create a custom XPath query to fetch the actual numeric value
|
||
|
if not propertyElem:
|
||
|
# Build xpath from version value, turn e.g. 'project.version' to '/pom:project/pom:version'
|
||
|
customXPath = ''.join(list(map(lambda x: '/pom:' + x, gavDictValue.split('.'))))
|
||
|
# Fetch the numeric version
|
||
|
propertyElem = getElementsByXPath(xmlRoot, customXPath)
|
||
|
# Exit with error if it wasn't possible to determine the artifact version even this way
|
||
|
_logErrorAndExitIf(
|
||
|
"Unable to determine the version of the '%s' GA artifact, exiting!" % gavDictKey,
|
||
|
len(propertyElem) != 1
|
||
|
)
|
||
|
# Assign the value of POM <property> or result of custom XPath
|
||
|
# back to 'gavDictValue' field and check again
|
||
|
gavDictValue = propertyElem[0].text
|
||
|
|
||
|
# Store the numeric artifact version into GAV dictionary, keeping
|
||
|
# the original POM <property> name as the key of the child dictionary
|
||
|
gavDictionary[gavDictKey] = { '%s' % childDictKey : '%s' % gavDictValue }
|
||
|
|
||
|
return gavDictionary
|
||
|
|
||
|
def mergeTwoGavDictionaries(firstGavDictionary, secondGavDictionary):
|
||
|
"""
|
||
|
Return a single output GAV dictionary containing the united content of
|
||
|
'firstGavDictionary' and 'secondGavDictionary' input GAV dictionaries.
|
||
|
|
||
|
The process of merge is performed as follows:
|
||
|
|
||
|
1) Distinct keys from both GAV dictionaries are copied into the output
|
||
|
dictionary.
|
||
|
|
||
|
2) If the key is present in both input GAV dictionaries (IOW it's shared),
|
||
|
the value of the higher version from both input dictionaries is used
|
||
|
as the final value for the united dictionary entry.
|
||
|
"""
|
||
|
unitedGavDictionary = copy.deepcopy(firstGavDictionary)
|
||
|
for secondDictKey in secondGavDictionary.keys():
|
||
|
try:
|
||
|
# Subcase when dictionary key from second GAV dictionary is
|
||
|
# already present in the resulting GAV dictionary
|
||
|
|
||
|
# Value of the key from resulting GAV dictionary might be a child
|
||
|
# dictionary again. Get the numeric version of the artifact first
|
||
|
currentValue = getNumericArtifactVersion(unitedGavDictionary, secondDictKey)
|
||
|
# Vaue of the key from second GAV dictionary might be a child
|
||
|
# dictionary again. Get the numeric version of the artifact first
|
||
|
secondDictValue = getNumericArtifactVersion(secondGavDictionary, secondDictKey)
|
||
|
|
||
|
# Update the artifact version in resulting GAV dictionary only if
|
||
|
# the value from the second dictionary is higher than the current
|
||
|
# one
|
||
|
if parseVersion(secondDictValue) > parseVersion(currentValue):
|
||
|
unitedGavDictionary[secondDictKey] = secondDictValue
|
||
|
|
||
|
except KeyError:
|
||
|
# Subcase when dictionary key from the second GAV dictionary is
|
||
|
# not present in the resulting GAV dictionary. Insert it
|
||
|
unitedGavDictionary[secondDictKey] = secondGavDictionary[secondDictKey]
|
||
|
|
||
|
return unitedGavDictionary
|
||
|
|
||
|
#
|
||
|
# Data structures and routines to assist with the updates of
|
||
|
# the main Keycloak pom.xml necessary for Wildfly upgrade
|
||
|
#
|
||
|
|
||
|
# List of artifacts from main Keycloak pom.xml excluded from upgrade even though they would
|
||
|
# be usually applicable for the update. This allows to handle special / corner case like for
|
||
|
# example the ones below:
|
||
|
#
|
||
|
# * The version / release tag of specific artifact, as used by upstream of that artifact is
|
||
|
# actually higher than the version, currently used in Wildfly / Wildfly Core. But the Python
|
||
|
# version comparing algorithm used by this script, treats it as a lower one
|
||
|
# (the cache of ApacheDS artifact below),
|
||
|
# * Explicitly avoid the update of certain artifact due whatever reason
|
||
|
#
|
||
|
# Add new entries to this list by moving them out of the _keycloakToWildflyProperties
|
||
|
# dictionary as necessary
|
||
|
_excludedProperties = [
|
||
|
# Intentionally avoid Apache DS downgrade from "2.0.0.AM26" to Wildfly's current
|
||
|
# "2.0.0-M24" version due to recent KEYCLOAK-14162
|
||
|
"apacheds.version"
|
||
|
]
|
||
|
|
||
|
# List of Keycloak specific properties listed in main Keycloak pom.xml file. These entries:
|
||
|
#
|
||
|
# * Either don't represent an artifact version (e.g. "product.rhsso.version" below),
|
||
|
# * Or represent an artifact version, but aren't used listed in Wildfly's or
|
||
|
# Wildfly-Core's POMs (the artifact is either not referenced in those POM files at all
|
||
|
# or explicitly excluded in some of them)
|
||
|
_keycloakSpecificProperties = [
|
||
|
"product.rhsso.version",
|
||
|
"product.build-time",
|
||
|
"eap.version",
|
||
|
"jboss.as.version",
|
||
|
"jboss.as.subsystem.test.version",
|
||
|
"jboss.aesh.version",
|
||
|
"jackson.databind.version",
|
||
|
"jackson.annotations.version",
|
||
|
"resteasy.undertow.version",
|
||
|
"owasp.html.sanitizer.version",
|
||
|
"sun.xml.ws.version",
|
||
|
"jetty92.version",
|
||
|
"jetty93.version",
|
||
|
"jetty94.version",
|
||
|
"ua-parser.version",
|
||
|
"version.com.openshift.openshift-restclient-java",
|
||
|
"apacheds.codec.version",
|
||
|
"google.zxing.version",
|
||
|
"freemarker.version",
|
||
|
"jetty9.version",
|
||
|
"liquibase.version",
|
||
|
"mysql.version",
|
||
|
"osgi.version",
|
||
|
"pax.web.version",
|
||
|
"postgresql.version",
|
||
|
"mariadb.version",
|
||
|
"mssql.version",
|
||
|
"twitter4j.version",
|
||
|
"jna.version",
|
||
|
"greenmail.version",
|
||
|
"jmeter.version",
|
||
|
"selenium.version",
|
||
|
"xml-apis.version",
|
||
|
"subethasmtp.version",
|
||
|
"replacer.plugin.version",
|
||
|
"jboss.as.plugin.version",
|
||
|
"jmeter.plugin.version",
|
||
|
"jmeter.analysis.plugin.version",
|
||
|
"minify.plugin.version",
|
||
|
"osgi.bundle.plugin.version",
|
||
|
"nexus.staging.plugin.version",
|
||
|
"frontend.plugin.version",
|
||
|
"docker.maven.plugin.version",
|
||
|
"surefire.memory.Xms",
|
||
|
"surefire.memory.Xmx",
|
||
|
"surefire.memory.metaspace",
|
||
|
"surefire.memory.metaspace.max",
|
||
|
"surefire.memory.settings",
|
||
|
"tomcat7.version",
|
||
|
"tomcat8.version",
|
||
|
"tomcat9.version",
|
||
|
"spring-boot15.version",
|
||
|
"spring-boot21.version",
|
||
|
"spring-boot22.version",
|
||
|
"webauthn4j.version",
|
||
|
"org.apache.kerby.kerby-asn1.version",
|
||
|
]
|
||
|
|
||
|
# Mapping of artifact name as used in the main Keycloak pom.xml file to the name
|
||
|
# of the same artifact listed in Wildfly's or Wildfly-Core's pom.xml file
|
||
|
_keycloakToWildflyProperties = {
|
||
|
"wildfly.version" : "version",
|
||
|
"wildfly.build-tools.version" : "version.org.wildfly.build-tools",
|
||
|
# Skip "eap.version" since Keycloak specific
|
||
|
"wildfly.core.version" : "version.org.wildfly.core",
|
||
|
# Skip "jboss.as.version" since Keycloak specific
|
||
|
# Skip "jboss.as.subsystem.test.version" since Keycloak specific
|
||
|
# Skip "jboss.aesh.version" since Keycloak specific
|
||
|
"aesh.version" : "version.org.aesh",
|
||
|
"apache.httpcomponents.version" : "version.org.apache.httpcomponents.httpclient",
|
||
|
"apache.httpcomponents.httpcore.version" : "version.org.apache.httpcomponents.httpcore",
|
||
|
"apache.mime4j.version" : "version.org.apache.james.apache-mime4j",
|
||
|
"jboss.dmr.version" : "version.org.jboss.jboss-dmr",
|
||
|
"bouncycastle.version" : "version.org.bouncycastle",
|
||
|
"cxf.version" : "version.org.apache.cxf",
|
||
|
"cxf.jetty.version" : "version.org.apache.cxf",
|
||
|
"cxf.jaxrs.version" : "version.org.apache.cxf",
|
||
|
"cxf.undertow.version" : "version.org.apache.cxf",
|
||
|
"dom4j.version" : "version.dom4j",
|
||
|
"h2.version" : "version.com.h2database",
|
||
|
"jakarta.persistence.version" : "version.jakarta.persistence",
|
||
|
"hibernate.core.version" : "version.org.hibernate",
|
||
|
"hibernate.c3p0.version" : "version.org.hibernate",
|
||
|
"infinispan.version" : "version.org.infinispan",
|
||
|
"jackson.version" : "version.com.fasterxml.jackson",
|
||
|
# Skip "jackson.databind.version" and "jackson.annotations.version" since they are derived from ${jackson.version}" above
|
||
|
"jakarta.mail.version" : "version.jakarta.mail",
|
||
|
"jboss.logging.version" : "version.org.jboss.logging.jboss-logging",
|
||
|
"jboss.logging.tools.version" : "version.org.jboss.logging.jboss-logging-tools",
|
||
|
"jboss-jaxrs-api_2.1_spec" : "version.org.jboss.spec.javax.ws.jboss-jaxrs-api_2.1_spec",
|
||
|
"jboss-transaction-api_1.3_spec" : "version.org.jboss.spec.javax.transaction.jboss-transaction-api_1.3_spec",
|
||
|
"jboss.spec.javax.xml.bind.jboss-jaxb-api_2.3_spec.version" : "version.org.jboss.spec.javax.xml.bind.jboss-jaxb-api_2.3_spec",
|
||
|
"jboss.spec.javax.servlet.jsp.jboss-jsp-api_2.3_spec.version" : "version.org.jboss.spec.javax.servlet.jsp.jboss-jsp-api_2.3_spec",
|
||
|
"log4j.version" : "version.log4j",
|
||
|
"resteasy.version" : "version.org.jboss.resteasy",
|
||
|
# Skip "resteasy.undertow.version" since it's derived from ${resteasy.version} above
|
||
|
# Skip "owasp.html.sanitizer.version" since Keycloak specific
|
||
|
"slf4j-api.version" : "version.org.slf4j",
|
||
|
"slf4j.version" : "version.org.slf4j",
|
||
|
"sun.istack.version" : "version.com.sun.istack",
|
||
|
"sun.xml.bind.version" : "version.sun.jaxb",
|
||
|
"javax.xml.bind.jaxb.version" : "version.javax.xml.bind.jaxb-api",
|
||
|
# Skip "sun.xml.ws.version" since Keycloak specific
|
||
|
"sun.activation.version" : "version.com.sun.activation.jakarta.activation",
|
||
|
"sun.xml.bind.version" : "version.sun.jaxb",
|
||
|
"org.glassfish.jaxb.xsom.version" : "version.sun.jaxb",
|
||
|
"undertow.version" : "version.io.undertow",
|
||
|
"elytron.version" : "version.org.wildfly.security.elytron",
|
||
|
"elytron.undertow-server.version" : "version.org.wildfly.security.elytron-web",
|
||
|
# Skip "jetty92.version", "jetty93.version", and "jetty94.version" since Keycloak specific
|
||
|
"woodstox.version" : "version.org.codehaus.woodstox.woodstox-core",
|
||
|
"xmlsec.version" : "version.org.apache.santuario",
|
||
|
"glassfish.json.version" : "version.org.glassfish.jakarta.json",
|
||
|
"wildfly.common.version" : "version.org.wildfly.common",
|
||
|
# Skip "ua-parser.version" since Keycloak specific
|
||
|
"picketbox.version" : "version.org.picketbox",
|
||
|
"google.guava.version" : "version.com.google.guava",
|
||
|
# Skip "version.com.openshift.openshift-restclient-java" since Keycloak specific
|
||
|
"commons-lang.version" : "version.commons-lang",
|
||
|
"commons-lang3.version" : "version.commons-lang3",
|
||
|
"commons-io.version" : "version.commons-io",
|
||
|
"apacheds.version" : "version.org.apache.ds",
|
||
|
# Skip "apacheds.codec.version" since Keycloak specific
|
||
|
# Skip "google.zxing.version" since Keycloak specific
|
||
|
# Skip "freemarker.version" since Keycloak specific
|
||
|
# Skip "jetty9.version" since Keycloak specific
|
||
|
# Skip "liquibase.version" since Keycloak specific
|
||
|
# Skip "mysql.version" since Keycloak specific
|
||
|
# Skip "osgi.version" since Keycloak specific
|
||
|
# Skip "pax.web.version" since Keycloak specific
|
||
|
# Skip "postgresql.version" since Keycloak specific
|
||
|
# Skip "mariadb.version" since Keycloak specific
|
||
|
# Skip "mssql.version" since Keycloak specific
|
||
|
"servlet.api.30.version" : "version.org.jboss.spec.javax.xml.soap.jboss-saaj-api_1.4_spec",
|
||
|
"servlet.api.40.version" : "version.org.jboss.spec.javax.servlet.jboss-servlet-api_4.0_spec",
|
||
|
# Skip "twitter4j.version" since Keycloak specific
|
||
|
# Skip "jna.version" since Keycloak specific
|
||
|
# Skip "greenmail.version" since Keycloak specific
|
||
|
"hamcrest.version" : "version.org.hamcrest",
|
||
|
# Skip "jmeter.version" since Keycloak specific
|
||
|
"junit.version" : "version.junit",
|
||
|
"picketlink.version" : "version.org.picketlink",
|
||
|
# Skip "selenium.version" since Keycloak specific
|
||
|
# Skip "xml-apis.version" since intentionally excluded in Wildfly
|
||
|
# Skip "subethasmtp.version" since Keycloak specific
|
||
|
"microprofile-metrics-api.version" : "version.org.eclipse.microprofile.metrics.api",
|
||
|
# Skip "replacer.plugin.version" since Keycloak specific
|
||
|
# Skip "jboss.as.plugin.version" since Keycloak specific
|
||
|
# Skip "jmeter.plugin.version" since Keycloak specific
|
||
|
# Skip "jmeter.analysis.plugin.version" since Keycloak specific
|
||
|
# Skip "minify.plugin.version" since Keycloak specific
|
||
|
# Skip "osgi.bundle.plugin.version" since Keycloak specific
|
||
|
"wildfly.plugin.version" : "version.org.wildfly.maven.plugins",
|
||
|
# Skip "nexus.staging.plugin.version" since Keycloak specific
|
||
|
# Skip "frontend.plugin.version" since Keycloak specific
|
||
|
# Skip "docker.maven.plugin.version" since Keycloak specific
|
||
|
# Skip "tomcat7.version", "tomcat8.version", and "tomcat9.version" since Keycloak specific
|
||
|
# Skip "spring-boot15.version", "spring-boot21.version", and "spring-boot22.version" since Keycloak specific
|
||
|
# Skip "webauthn4j.version" since Keycloak specific
|
||
|
# Skip "org.apache.kerby.kerby-asn1.version" since Keycloak specific
|
||
|
}
|
||
|
|
||
|
def _scanMainKeycloakPomFileForUnknownArtifacts():
|
||
|
"""
|
||
|
Verify each artifact listed as property in the main Keycloak pom.xml file is present one of the:
|
||
|
|
||
|
* _excludedProperties list -- explicitly requesting the update to be skipped due some reason,
|
||
|
* _keycloakSpecificProperties list -- artifact is Keycloak specific,
|
||
|
* _keycloakToWildflyProperties dictionary -- there's a clear mapping of Keycloak
|
||
|
artifact property name to corresponding artifact property name as used in Wildfly /
|
||
|
Wildfly Core
|
||
|
|
||
|
Logs error message and exits with error if action for a particular artifact is unknown.
|
||
|
"""
|
||
|
# Absolute path to main Keycloak pom.xml within the repo
|
||
|
mainKeycloakPomPath = getKeycloakGitRepositoryRoot() + "/pom.xml"
|
||
|
|
||
|
unknownArtifactMessage = (
|
||
|
"Found so far unknown '%s' artifact in the main Keycloak pom.xml file!\n"
|
||
|
"There's no clearly defined action on how to process this artifact yet!\n"
|
||
|
"It's not an excluded one, not listed as Keycloak specific one, and not\n"
|
||
|
"present in the set of those to be processed. Add it to one of:\n\n"
|
||
|
" * _excludedProperties,\n"
|
||
|
" * _keycloakSpecificProperties,\n"
|
||
|
" * or _keycloakToWildflyProperties \n\n"
|
||
|
"data structures in \"wildfly/upgrade/__init__.py\" to dismiss this error!\n"
|
||
|
"Rerun the script once done."
|
||
|
)
|
||
|
for xmlTag in getElementsByXPath(getXmlRoot(mainKeycloakPomPath), '//pom:project/pom:properties/pom:*'):
|
||
|
artifactName = xmlTag.tag.replace("{%s}" % _pom_ns, "")
|
||
|
_logErrorAndExitIf (
|
||
|
unknownArtifactMessage % artifactName,
|
||
|
artifactName not in itertools.chain(_excludedProperties, _keycloakSpecificProperties, _keycloakToWildflyProperties.keys())
|
||
|
)
|
||
|
|
||
|
# Empirical list of artifacts to retrieve from Wildfly-Core's pom.xml rather than from Wildfly's pom.xml
|
||
|
_wildflyCoreProperties = [
|
||
|
"wildfly.build-tools.version",
|
||
|
"aesh.version",
|
||
|
"apache.httpcomponents.version",
|
||
|
"apache.httpcomponents.httpcore.version",
|
||
|
"jboss.dmr.version",
|
||
|
"jboss.logging.version",
|
||
|
"jboss.logging.tools.version",
|
||
|
"log4j.version",
|
||
|
"slf4j-api.version",
|
||
|
"slf4j.version",
|
||
|
"javax.xml.bind.jaxb.version",
|
||
|
"undertow.version",
|
||
|
"elytron.version",
|
||
|
"elytron.undertow-server.version",
|
||
|
"woodstox.version",
|
||
|
"glassfish.json.version",
|
||
|
"picketbox.version",
|
||
|
"commons-lang.version",
|
||
|
"commons-io.version",
|
||
|
"junit.version",
|
||
|
]
|
||
|
|
||
|
def updateMainKeycloakPomFile(wildflyPomFile, wildflyCorePomFile):
|
||
|
"""
|
||
|
Synchronize the versions of artifacts listed as properties in the main
|
||
|
Keycloak pom.xml file with their counterparts taken from 'wildflyPomFile'
|
||
|
and 'wildflyCorePomFile'.
|
||
|
"""
|
||
|
wildflyXmlTreeRoot = getXmlRoot(wildflyPomFile)
|
||
|
wildflyCoreXmlTreeRoot = getXmlRoot(wildflyCorePomFile)
|
||
|
|
||
|
# Absolute path to main Keycloak pom.xml within the repo
|
||
|
mainKeycloakPomPath = getKeycloakGitRepositoryRoot() + "/pom.xml"
|
||
|
keycloakXmlTreeRoot = getXmlRoot(mainKeycloakPomPath)
|
||
|
|
||
|
taskLogger = getTaskLogger('Update main Keycloak pom.xml')
|
||
|
taskLogger.info('Synchronizing Wildfly (Core) artifact versions to the main Keycloak pom.xml file...')
|
||
|
|
||
|
stepLogger = getStepLogger()
|
||
|
|
||
|
_scanMainKeycloakPomFileForUnknownArtifacts()
|
||
|
|
||
|
for keycloakElemName, wildflyElemName in _keycloakToWildflyProperties.items():
|
||
|
|
||
|
if keycloakElemName == "wildfly.version":
|
||
|
wildflyElem = getElementsByXPath(wildflyXmlTreeRoot, '/pom:project/pom:version')
|
||
|
# Artifact is one of those listed above to be fetched from Wildfly Core's pom.xml
|
||
|
elif keycloakElemName in _wildflyCoreProperties:
|
||
|
wildflyElem = getPomProperty(wildflyCoreXmlTreeRoot, wildflyElemName)
|
||
|
# Otherwise fetch artifact version from Wildfly's pom.xml
|
||
|
else:
|
||
|
wildflyElem = getPomProperty(wildflyXmlTreeRoot, wildflyElemName)
|
||
|
|
||
|
if wildflyElem:
|
||
|
keycloakElem = getPomProperty(keycloakXmlTreeRoot, keycloakElemName)
|
||
|
if keycloakElem:
|
||
|
if keycloakElemName in _excludedProperties:
|
||
|
stepLogger.debug(
|
||
|
"Not updating version of '%s' from '%s' to '%s' since the artifact is excluded!" %
|
||
|
(keycloakElemName, keycloakElem[0].text, wildflyElem[0].text)
|
||
|
)
|
||
|
elif parseVersion(wildflyElem[0].text) > parseVersion(keycloakElem[0].text):
|
||
|
stepLogger.debug(
|
||
|
"Updating version of '%s' artifact to '%s'. Current '%s' version is less than that." %
|
||
|
(keycloakElemName, wildflyElem[0].text, keycloakElem[0].text)
|
||
|
)
|
||
|
keycloakElem[0].text = wildflyElem[0].text
|
||
|
else:
|
||
|
stepLogger.debug(
|
||
|
"Not updating version of '%s' artifact to '%s'. Current '%s' version is already up2date." %
|
||
|
(keycloakElemName, wildflyElem[0].text, keycloakElem[0].text)
|
||
|
)
|
||
|
else:
|
||
|
stepLogger.error(
|
||
|
"Unable to locate element with name: '%s' in '%s' or '%s'" %
|
||
|
(wildflyElemName, wildflyPomFile, wildflyCorePomFile)
|
||
|
)
|
||
|
|
||
|
lxml.etree.ElementTree(keycloakXmlTreeRoot).write(mainKeycloakPomPath, encoding = "UTF-8", pretty_print = True, xml_declaration = True)
|
||
|
stepLogger.info("Done syncing artifact version changes to: '%s'" % mainKeycloakPomPath.replace(getKeycloakGitRepositoryRoot(), '.'))
|
||
|
stepLogger.debug("Wrote updated main Keycloak pom.xml file to: '%s'" % mainKeycloakPomPath)
|
||
|
|
||
|
#
|
||
|
# Routing handling necessary updates of various
|
||
|
# adapter license files related with a Wildfly upgrade
|
||
|
#
|
||
|
|
||
|
def updateAdapterLicenseFile(gavDictionary, xPathPrefix, nameSpace, licenseFile):
|
||
|
"""
|
||
|
Save GAV dictionary 'gavDictionary' back to XML 'licenseFile'.
|
||
|
"""
|
||
|
licenseFileXmlTreeRoot = getXmlRoot(licenseFile)
|
||
|
LICENSE_FILE_PARENT_DIR = os.path.dirname(licenseFile)
|
||
|
stepLogger = getStepLogger()
|
||
|
|
||
|
if not nameSpace:
|
||
|
nsPrefix = ''
|
||
|
dependencyElemXPath = '|'.join(map(lambda e: xPathPrefix + '/%s' % e, _gav_elements))
|
||
|
else:
|
||
|
nsPrefix = nameSpace.keys()
|
||
|
dependencyElemXPath = '|'.join(map(lambda e: xPathPrefix + '/%s:%s' % (nsPrefix, e), _gav_elements))
|
||
|
|
||
|
xmlDependencyElements = getElementsByXPath(licenseFileXmlTreeRoot, dependencyElemXPath, nameSpace)
|
||
|
# Divide original list into sublists by three elements -- one sublist per GAV entry
|
||
|
for gavEntry in [xmlDependencyElements[i:i + 3] for i in range(0, len(xmlDependencyElements), 3)]:
|
||
|
currentArtifactVersion = expectedArtifactVersion = None
|
||
|
groupIdElem, artifactIdElem, versionElem = gavEntry[0], gavEntry[1], gavEntry[2]
|
||
|
_logErrorAndExitIf(
|
||
|
"Failed to update '%s' XML dependency!" % gavEntry,
|
||
|
groupIdElem is None or artifactIdElem is None or versionElem is None
|
||
|
)
|
||
|
currentArtifactVersion = versionElem.text
|
||
|
gavDictKey = groupIdElem.text + _gav_delimiter + artifactIdElem.text
|
||
|
try:
|
||
|
# Value of the artifact version might be a child dictionary again.
|
||
|
# Get numeric artifact version first
|
||
|
expectedArtifactVersion = getNumericArtifactVersion(gavDictionary, gavDictKey)
|
||
|
# Update the version of artifact if version from GAV dictionary is higher
|
||
|
if expectedArtifactVersion and parseVersion(expectedArtifactVersion) > parseVersion(versionElem.text):
|
||
|
updatingArtifactVersionMessage = (
|
||
|
"Updating the version of '%s, %s' artifact in license file from: '%s' to: '%s'" %
|
||
|
(groupIdElem.text, artifactIdElem.text, currentArtifactVersion, expectedArtifactVersion)
|
||
|
)
|
||
|
stepLogger.debug(updatingArtifactVersionMessage)
|
||
|
versionElem.text = expectedArtifactVersion
|
||
|
# Subtask: Rename existing license text files tracked in this repository to the filename with the updated artifact version
|
||
|
repositoryRoot = getKeycloakGitRepositoryRoot()
|
||
|
for root, dirs, files in os.walk(LICENSE_FILE_PARENT_DIR):
|
||
|
for filename in files:
|
||
|
if re.search(re.escape(artifactIdElem.text) + r',' + re.escape(currentArtifactVersion), filename):
|
||
|
currentFilename = filename
|
||
|
currentFileName = currentFilename.replace(repositoryRoot, '').rstrip()
|
||
|
newFilename = currentFilename.replace(currentArtifactVersion, expectedArtifactVersion)
|
||
|
check_call(['git', 'mv', "%s" % os.path.join(root, currentFilename), "%s" % os.path.join(root, newFilename)], cwd = repositoryRoot)
|
||
|
# Subtask: Update artifact version in license URL to the expected one
|
||
|
dependencyElem = groupIdElem.getparent()
|
||
|
urlElements = getElementsByXPath(dependencyElem, './licenses/license/url', nameSpace)
|
||
|
_logErrorAndExitIf(
|
||
|
"Failed to retrieve <url> element of the '%s' artifact!" % gavDictKey,
|
||
|
len(urlElements) != 1
|
||
|
)
|
||
|
urlElem = urlElements[0]
|
||
|
# Strip the '.redhat-\d+' suffix from artifact versions when processing RH-SSO adapters
|
||
|
# since upstream URLs don't contain those
|
||
|
if 'rh-sso' in licenseFile:
|
||
|
expectedArtifactVersion = re.sub(r'.redhat-\d+$', '', expectedArtifactVersion)
|
||
|
# First handle special form of version numbers in release URLs used by org.bouncycastle artifacts
|
||
|
if artifactIdElem.text.endswith('jdk15on'):
|
||
|
bouncyCastleMajorVersion = re.match(r'^(\d)\.', expectedArtifactVersion).group(1)
|
||
|
bouncyCastleMinorVersion = re.match(r'^\d+\.(\d+)', expectedArtifactVersion).group(1)
|
||
|
if bouncyCastleMajorVersion and bouncyCastleMinorVersion:
|
||
|
urlNotationOfExpectedBouncyCastleVersion = 'r' + bouncyCastleMajorVersion + 'rv' + bouncyCastleMinorVersion
|
||
|
try:
|
||
|
# Extract older (even archaic) 'major.minor.micro' artifact version substring from the URL
|
||
|
oldMajorMinorMicroVersion = re.search(r'(r\d+rv\d{2,})', urlElem.text).group(1)
|
||
|
if oldMajorMinorMicroVersion:
|
||
|
stepLogger.debug(
|
||
|
"Replacing former '%s' of '%s' artifact version in the URL with the new '%s' version" %
|
||
|
(oldMajorMinorMicroVersion, gavDictKey, expectedArtifactVersion)
|
||
|
)
|
||
|
urlElem.text = re.sub(r'r\d+rv\d{2,}', urlNotationOfExpectedBouncyCastleVersion, urlElem.text)
|
||
|
except AttributeError:
|
||
|
# Ignore generic URLs not containing 'major.minor.micro' information of this specific artifact
|
||
|
pass
|
||
|
else:
|
||
|
_logErrorAndExitIf(
|
||
|
"Unable to locate previous '%s' artifact version in the URL!" % gavDictKey,
|
||
|
True
|
||
|
)
|
||
|
else:
|
||
|
try:
|
||
|
# Extract older (even archaic) 'major.minor.micro' artifact version substring from the URL
|
||
|
oldMajorMinorMicroVersion = re.search(r'(\d+\.\d+\.\d+)', urlElem.text).group(1)
|
||
|
if oldMajorMinorMicroVersion:
|
||
|
stepLogger.debug(
|
||
|
"Replacing former '%s' version of the '%s' artifact in the URL with the new '%s' version" %
|
||
|
(oldMajorMinorMicroVersion, gavDictKey, expectedArtifactVersion)
|
||
|
)
|
||
|
urlElem.text = re.sub(oldMajorMinorMicroVersion, expectedArtifactVersion, urlElem.text)
|
||
|
else:
|
||
|
_logErrorAndExitIf(
|
||
|
"Unable to locate previous '%s' artifact version in the URL!" % gavDictKey,
|
||
|
True
|
||
|
)
|
||
|
except AttributeError:
|
||
|
# Ignore generic URLs not containing 'major.minor.micro' information of this specific artifact
|
||
|
pass
|
||
|
else:
|
||
|
artifactVersionAlreadyHigherMessage = (
|
||
|
"Not updating version of '%s, %s' artifact to '%s'. Current '%s' version is already up2date." %
|
||
|
(groupIdElem.text, artifactIdElem.text, expectedArtifactVersion, currentArtifactVersion)
|
||
|
)
|
||
|
stepLogger.debug(artifactVersionAlreadyHigherMessage)
|
||
|
|
||
|
except KeyError:
|
||
|
# Ignore artifacts not found in the Gav dictionary
|
||
|
stepLogger.debug("Skipping '%s' artifact not present in GAV dictionary." % gavDictKey)
|
||
|
pass
|
||
|
|
||
|
lxml.etree.ElementTree(licenseFileXmlTreeRoot).write(licenseFile, encoding = "UTF-8", pretty_print = True, xml_declaration = True)
|
||
|
relativeLicenseFilePath = licenseFile.replace(getKeycloakGitRepositoryRoot(), '.')
|
||
|
stepLogger.info("Done syncing artifact version changes to: '%s'" % relativeLicenseFilePath)
|
||
|
stepLogger.debug("Wrote updated license file to: '%s'" % licenseFile)
|
||
|
|
||
|
#
|
||
|
# Routines performing particular tasks within a Wildfly upgrade
|
||
|
#
|
||
|
|
||
|
def performKeycloakAdapterLicenseFilesUpdateTask(wildflyPomFile, wildflyCorePomFile):
|
||
|
"""
|
||
|
Update artifacts versions of selected dependencies utilized by various
|
||
|
Keycloak adapter license XML files. Also update the location of the
|
||
|
corresponding license text files within the repository so their names
|
||
|
reflect the updated artifacts versions.
|
||
|
"""
|
||
|
# Operate on Keycloak adapters
|
||
|
PROFILE = 'community'
|
||
|
|
||
|
# Load XML dependencies from Wildfly (Core) POM files into GAV dictionary
|
||
|
wildflyCoreXmlDependenciesGav = loadGavDictionaryFromXmlFile(wildflyCorePomFile)
|
||
|
wildflyXmlDependenciesGav = loadGavDictionaryFromXmlFile(wildflyPomFile)
|
||
|
# Merge both Wildfly and Wildfly Core GAV dictionaries into a united one,
|
||
|
# containing all Wildfly (Core) artifacts and their versions
|
||
|
unitedGavDictionary = mergeTwoGavDictionaries(
|
||
|
wildflyCoreXmlDependenciesGav,
|
||
|
wildflyXmlDependenciesGav
|
||
|
)
|
||
|
|
||
|
isTaskLogged = False
|
||
|
(productName, productNameFull) = getProductNamesForKeycloakPomProfile(profile = PROFILE)
|
||
|
taskLogger = getTaskLogger('Update %s Adapters' % productNameFull)
|
||
|
gitRepositoryRoot = getKeycloakGitRepositoryRoot()
|
||
|
for root, dirs, files in os.walk(gitRepositoryRoot):
|
||
|
if not isTaskLogged:
|
||
|
taskLabel = (
|
||
|
"Updating artifacts versions in license XML files and locations of the license TXT files" +
|
||
|
"\n\tfor the %s adapters in the '%s' directory..." % (productName, root)
|
||
|
)
|
||
|
taskLogger.info(taskLabel)
|
||
|
isTaskLogged = True
|
||
|
for filename in files:
|
||
|
if re.search(r'distribution.*%s.*licenses.xml' % productName.lower(), os.path.join(root, filename)):
|
||
|
updateAdapterLicenseFile(
|
||
|
unitedGavDictionary,
|
||
|
xPathPrefix = '/licenseSummary/dependencies/dependency',
|
||
|
nameSpace = {},
|
||
|
licenseFile = os.path.join(root, filename)
|
||
|
)
|
||
|
|
||
|
def performRhssoAdapterLicenseFilesUpdateTask(wildflyPomFile, wildflyCorePomFile):
|
||
|
"""
|
||
|
Update artifacts versions of selected dependencies utilized by various
|
||
|
RH-SSO adapter license XML files. Also update the location of the
|
||
|
corresponding license text files within the repository so their names
|
||
|
reflect the updated artifacts versions.
|
||
|
"""
|
||
|
# Operate on RH-SSO adapters
|
||
|
PROFILE = 'product'
|
||
|
|
||
|
isTaskLogged = False
|
||
|
(productName, productNameFull) = getProductNamesForKeycloakPomProfile(profile = PROFILE)
|
||
|
taskLogger = getTaskLogger('Update %s Adapters' % productNameFull)
|
||
|
|
||
|
gavFileUrl = None
|
||
|
print("\nPlease specify the URL of the GAV file to use for %s adapter updates:" % productNameFull.upper())
|
||
|
gavFileUrl = sys.stdin.readline().rstrip()
|
||
|
|
||
|
_logErrorAndExitIf(
|
||
|
"Invalid URL '%s'! Please provide valid URL to the GAV file and retry!" % gavFileUrl,
|
||
|
not gavFileUrl or not gavFileUrl.startswith('http://') and not gavFileUrl.startswith('https://')
|
||
|
)
|
||
|
gavFile = saveUrlToNamedTemporaryFile(gavFileUrl)
|
||
|
taskLogger.debug("Downloaded content of provided GAV file to '%s'" % gavFile)
|
||
|
gavDictionary = loadGavDictionaryFromGavFile(gavFile)
|
||
|
|
||
|
gitRepositoryRoot = getKeycloakGitRepositoryRoot()
|
||
|
for root, dirs, files in os.walk(gitRepositoryRoot):
|
||
|
if not isTaskLogged:
|
||
|
taskLabel = (
|
||
|
"Updating artifacts versions in license XML files and locations of the license TXT files" +
|
||
|
"\n\tfor the %s adapters in the '%s' directory..." % (productName.upper(), root)
|
||
|
)
|
||
|
taskLogger.info(taskLabel)
|
||
|
isTaskLogged = True
|
||
|
for filename in files:
|
||
|
if re.search(r'distribution.*%s.*licenses.xml' % productName.lower(), os.path.join(root, filename)):
|
||
|
updateAdapterLicenseFile(
|
||
|
gavDictionary,
|
||
|
xPathPrefix = '/licenseSummary/dependencies/dependency',
|
||
|
nameSpace = {},
|
||
|
licenseFile = os.path.join(root, filename)
|
||
|
)
|