You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
194 lines
7.2 KiB
194 lines
7.2 KiB
# @HEADER
|
|
# ************************************************************************
|
|
#
|
|
# TriBITS: Tribal Build, Integrate, and Test System
|
|
# Copyright 2013 Sandia Corporation
|
|
#
|
|
# Under the terms of Contract DE-AC04-94AL85000 with Sandia Corporation,
|
|
# the U.S. Government retains certain rights in this software.
|
|
#
|
|
# Redistribution and use in source and binary forms, with or without
|
|
# modification, are permitted provided that the following conditions are
|
|
# met:
|
|
#
|
|
# 1. Redistributions of source code must retain the above copyright
|
|
# notice, this list of conditions and the following disclaimer.
|
|
#
|
|
# 2. Redistributions in binary form must reproduce the above copyright
|
|
# notice, this list of conditions and the following disclaimer in the
|
|
# documentation and/or other materials provided with the distribution.
|
|
#
|
|
# 3. Neither the name of the Corporation nor the names of the
|
|
# contributors may be used to endorse or promote products derived from
|
|
# this software without specific prior written permission.
|
|
#
|
|
# THIS SOFTWARE IS PROVIDED BY SANDIA CORPORATION "AS IS" AND ANY
|
|
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL SANDIA CORPORATION OR THE
|
|
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
|
|
# EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
# PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
# PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
# LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
# NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
#
|
|
# ************************************************************************
|
|
# @HEADER
|
|
|
|
|
|
#
|
|
# General scripting support
|
|
#
|
|
# NOTE: Included first to check the version of python!
|
|
#
|
|
|
|
from TribitsDependencies import getProjectDependenciesFromXmlFile
|
|
from GeneralScriptSupport import *
|
|
import time
|
|
|
|
|
|
#
|
|
# Default logic to determine if a changed file should trigger testing all of
|
|
# the packages.
|
|
#
|
|
# In general, any file at the top level <projectDir>/ or under the
|
|
# <projectDir>/cmake/ directory with the extension *.cmake is considered a
|
|
# file that requires a global rebuild of all packaes. However, there are a
|
|
# few special files that we don't want to have to do a global rebuild for by
|
|
# default.
|
|
#
|
|
# PackagesList.cmake: This file gets modified frequently to add new packages
|
|
# and rearrange packages. We don't need to do a global rebuild because this
|
|
# list of packages is validated if we do even a single rebuild. If a package
|
|
# line gets removed, the code that reads the Dependencies.cmake files will
|
|
# fail and stop. However, this has the risk that if only the test test
|
|
# category changes (e.g. from 'ST' to 'PT') then this would not trigger a
|
|
# rebuild. But if the package itself was also modified, then that would be
|
|
# sufficient for testing.
|
|
#
|
|
# TPLsList.cmake: This file also gets modified frequently. We don't need to
|
|
# enable all packages either for the same reason as for the PackagesList.cmake
|
|
# file.
|
|
#
|
|
# cmake/ctest/: These drive nightly automated builds and should not require
|
|
# testing every package.
|
|
#
|
|
# cmake/TPLs/*.cmake: Any FileTPLSOMETHING.cmake file is assumed not to be
|
|
# needed for Primary Tested code and therefore does not need to trigger a
|
|
# global rebuild.
|
|
#
|
|
|
|
class DefaultProjectCiFileChangeLogic:
|
|
|
|
def isGlobalBuildFileRequiringGlobalRebuild(self, modifiedFileFullPath):
|
|
modifiedFileFullPathArray = getFilePathArray(modifiedFileFullPath)
|
|
if len(modifiedFileFullPathArray)==1:
|
|
# Files sitting directly under <projectDir>/
|
|
if modifiedFileFullPathArray[0] == "CMakeLists.txt":
|
|
return True
|
|
if modifiedFileFullPathArray[0] == 'PackagesList.cmake':
|
|
return False
|
|
if modifiedFileFullPathArray[0] == 'TPLsList.cmake':
|
|
return False
|
|
if modifiedFileFullPathArray[0].rfind(".cmake") != -1:
|
|
return True
|
|
elif modifiedFileFullPathArray[0] == 'cmake':
|
|
# Files under <projectDir>/cmake/
|
|
if modifiedFileFullPathArray[1]=='ExtraRepositoriesList.cmake':
|
|
return False
|
|
if modifiedFileFullPathArray[1] == 'ctest':
|
|
return False
|
|
if modifiedFileFullPathArray[1] == 'TPLs':
|
|
return False
|
|
if modifiedFileFullPath.rfind("UnitTests/") != -1:
|
|
return False
|
|
if modifiedFileFullPath.rfind(".cmake") != -1:
|
|
return True
|
|
return False
|
|
|
|
|
|
def getProjectCiFileChangeLogic(projectDir):
|
|
|
|
if not projectDir:
|
|
return DefaultProjectCiFileChangeLogic()
|
|
else:
|
|
projectCiFileChangeLogicFileBaseDir = os.path.join(projectDir, "cmake")
|
|
projectCiFileChangeLogicFile = os.path.join(
|
|
projectCiFileChangeLogicFileBaseDir, "ProjectCiFileChangeLogic.py")
|
|
if not os.path.isfile(projectCiFileChangeLogicFile):
|
|
return DefaultProjectCiFileChangeLogic()
|
|
|
|
# Else, if we get here, then the file
|
|
# <projectDir>/cmake/ProjectCiFileChangeLogic.py exists so lets read it in!
|
|
|
|
tribitsDirPath = os.path.abspath(
|
|
os.path.join(
|
|
os.path.dirname(os.path.abspath(__file__)),
|
|
"../..", "tribits"
|
|
)
|
|
)
|
|
|
|
old_sys_path = sys.path
|
|
|
|
try:
|
|
sys.path = [projectCiFileChangeLogicFileBaseDir] + sys.path
|
|
import ProjectCiFileChangeLogic
|
|
return ProjectCiFileChangeLogic.ProjectCiFileChangeLogic()
|
|
finally:
|
|
sys.path = old_sys_path
|
|
|
|
|
|
def getPackageStructFromPath(projectDependencies, fullPath):
|
|
packageName = getPackageNameFromPath(projectDependencies, fullPath)
|
|
if packageName:
|
|
return projectDependencies.getPackageByName(packageName)
|
|
return None
|
|
|
|
|
|
def getPackageNameFromPath(projectDependencies, fullPath):
|
|
return projectDependencies.getPackageNameFromPath(fullPath)
|
|
|
|
|
|
def extractFilesListMatchingPattern(fileList_in, reMatachingPattern):
|
|
fileList_out = []
|
|
for line in fileList_in:
|
|
reFilePatternMatch = reMatachingPattern.match(line)
|
|
if reFilePatternMatch:
|
|
fileList_out.append(reFilePatternMatch.group(1).strip())
|
|
return fileList_out
|
|
|
|
|
|
def getPackagesListFromFilePathsList(projectDependencies, filePathsList,
|
|
allPackages=False, projectCiFileChangeLogic=DefaultProjectCiFileChangeLogic() \
|
|
):
|
|
packagesList = []
|
|
enabledAllPackages = False
|
|
for filePath in filePathsList:
|
|
packageName = getPackageNameFromPath(projectDependencies, filePath)
|
|
if findInSequence(packagesList, packageName) == -1 and packageName:
|
|
packagesList.append(packageName.strip())
|
|
if allPackages \
|
|
and projectCiFileChangeLogic.isGlobalBuildFileRequiringGlobalRebuild(filePath) \
|
|
and not enabledAllPackages \
|
|
:
|
|
packagesList.append("ALL_PACKAGES")
|
|
enabledAllPackages = True
|
|
return packagesList
|
|
|
|
|
|
def getPackageCheckinEmailAddressesListFromFilePathsList(
|
|
projectDependencies, filePathsList \
|
|
) \
|
|
:
|
|
packageCheckinEmailAddresses = []
|
|
for filePath in filePathsList:
|
|
packageStruct = getPackageStructFromPath(projectDependencies, filePath)
|
|
if packageStruct:
|
|
checkinEmail = packageStruct.emailAddresses.checkin
|
|
else:
|
|
checkinEmail = None
|
|
if findInSequence(packageCheckinEmailAddresses, checkinEmail) == -1:
|
|
packageCheckinEmailAddresses.append(checkinEmail)
|
|
return packageCheckinEmailAddresses
|
|
|