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
 | |
| 
 |