All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.spdx.merge.SpdxPackageInfoMerger Maven / Gradle / Ivy

There is a newer version: 2.2.8
Show newest version
/**
 * Copyright (c) 2014 Gang Ling.
 *
 *   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.
 *
*/
package org.spdx.merge;

import java.security.NoSuchAlgorithmException;
import java.util.ArrayList;
import java.util.List;

import org.spdx.rdfparser.InvalidSPDXAnalysisException;
import org.spdx.rdfparser.JavaSha1ChecksumGenerator;
import org.spdx.rdfparser.SpdxPackageVerificationCode;
import org.spdx.rdfparser.VerificationCodeGenerator;
import org.spdx.rdfparser.license.AnyLicenseInfo;
import org.spdx.rdfparser.license.ExtractedLicenseInfo;
import org.spdx.rdfparser.license.LicenseInfoFactory;
import org.spdx.rdfparser.model.SpdxDocument;
import org.spdx.rdfparser.model.SpdxFile;
import org.spdx.rdfparser.model.SpdxPackage;
import org.spdx.spdxspreadsheet.InvalidLicenseStringException;

import com.google.common.collect.Lists;

/**
 * Application to merge package information from input SPDX documents and file information merging result.
 * @author Gang Ling
 *
 */
public class SpdxPackageInfoMerger {

	    @Deprecated
		private SpdxPackage packageInfoResult = null;
		
		private List packagesResult = null;
		private SpdxDocument[] subDocs = null;
		private SpdxLicenseMapper mapper = null;
		
		/** 
		 * @param masterPackagesInfo
		 * @param subDocs
		 */
		public SpdxPackageInfoMerger(List masterPackagesInfo, SpdxDocument[] subDocs
				, SpdxLicenseMapper mapper){
			this.packagesResult = masterPackagesInfo;
			this.subDocs = subDocs.clone();
			this.mapper = mapper;
		}
		
		/** A method to merge all packages' information from sub list documents into master document
		 * @param subDocs
		 * @param fileMergeResult
		 * @return packagesResult
		 * @throws InvalidSPDXAnalysisException
		 */
		public List  mergePackagesInfo(SpdxFile[] fileMergeResult)
				throws InvalidSPDXAnalysisException, NoSuchAlgorithmException, InvalidLicenseStringException{
			
			List retval = Lists.newArrayList(clonePackages(packagesResult));
			
			for(int i = 0; i < subDocs.length; i++){				
				List subPackagesInfo = subDocs[i].getDocumentContainer().findAllPackages();
				SpdxPackage tempPackage = null;
				for(int p = 0; p < subPackagesInfo.size(); p++){
					boolean foundNameMatch = false;
					boolean foundSha1Match = false;
					SpdxPackage masterPackage = null;
					Integer index = null;
	                tempPackage = subPackagesInfo.get(p);
	                
	                for(int q = 0; q < retval.size(); q++){
	                	if(retval.get(q).getName().equalsIgnoreCase(tempPackage.getName())) {
                            foundNameMatch = true;
                        }
	                	if(retval.get(q).getSha1().equals(tempPackage.getSha1())){
	                		foundSha1Match = true;
	                		masterPackage = retval.get(q);
	                		index = q;
	                		break;
	                	}
	                }
	                if(!foundNameMatch && !foundSha1Match){
	                	AnyLicenseInfo[] licFromFile = checkLicenseFromFile(subDocs[i],tempPackage);
	                	tempPackage.setLicenseInfosFromFiles(licFromFile);
	                	retval.add(tempPackage);
	                }
	                else{
	                	//process to merge license info from file
	                	 AnyLicenseInfo[] masterLicFromFile = masterPackage.getLicenseInfoFromFiles();
	                	 AnyLicenseInfo[] licFromFile = checkLicenseFromFile(subDocs[i],tempPackage);
	                	 ArrayList  licList = Lists.newArrayList(masterLicFromFile);
	                	 for(int g = 0; g < licFromFile.length; g++){
	                		 for(int d = 0; d < licList.size(); d++){
	                			 if(licList.get(d).equals(licFromFile[g])) {
                                    break;
                                } else {
                                    licList.add(licFromFile[g]);
                                }
	                		 }
	                	 }
	                	 AnyLicenseInfo[] mergedLicFromFile = licList.toArray(new AnyLicenseInfo[licList.size()]);
	                	 masterPackage.setLicenseInfosFromFiles(mergedLicFromFile);
	                	 
	                	 //process to generate new verification code
	                	 String[] skippedFiles = collectSkippedFiles(masterPackage,tempPackage);
	                	 VerificationCodeGenerator verCodeGenerator = new VerificationCodeGenerator(new JavaSha1ChecksumGenerator());
	                	 SpdxPackageVerificationCode newCode = verCodeGenerator.generatePackageVerificationCode(fileMergeResult, skippedFiles);
	                	 masterPackage.setPackageVerificationCode(newCode);
	                	 
	                	 //process to merge package supplier info
	                	 String supplierInfo = stringCombiner(masterPackage.getSupplier(),tempPackage.getSupplier());
	                	 masterPackage.setSupplier(supplierInfo);
	                	 
	                	 //process to merge package originator
	                	 String originator = stringCombiner(masterPackage.getOriginator(), tempPackage.getOriginator());
	                	 masterPackage.setOriginator(originator);
	                	 
	                	 //process to merge home page
	                	 String homePage = stringCombiner(masterPackage.getHomepage(), tempPackage.getHomepage());
	                	 masterPackage.setHomepage(homePage);
	                	 
	                	 //process to merge package download location
	                	 String downloadLocation = stringCombiner(masterPackage.getDownloadLocation(), tempPackage.getDownloadLocation());
	                	 masterPackage.setDownloadLocation(downloadLocation);
	                	 
	                	 //process to merge source info
	                	 String sourceInfo = stringCombiner(masterPackage.getSourceInfo(), tempPackage.getSourceInfo());
	                	 masterPackage.setSourceInfo(sourceInfo);
	                	 
	                	 //process to merge license declared
	                	 AnyLicenseInfo licDeclared1 = masterPackage.getLicenseDeclared();
	                	 AnyLicenseInfo licDeclared2 = mapper.mapLicenseInfo(subDocs[i], tempPackage.getLicenseDeclared());
	                	 AnyLicenseInfo licDeclared = licsCombiner(licDeclared1, licDeclared2);
	                	 masterPackage.setLicenseDeclared(licDeclared);
	                	 
	                	 //process to merge license concluded
	                	 AnyLicenseInfo licConcluded1 = masterPackage.getLicenseConcluded();
	                	 AnyLicenseInfo licConcluded2 = mapper.mapLicenseInfo(subDocs[i], tempPackage.getLicenseConcluded());
	                	 AnyLicenseInfo licConcluded = licsCombiner(licConcluded1, licConcluded2);
	                	 masterPackage.setLicenseConcluded(licConcluded);
	                	 
	                	 //process to merge license comments
	                	 String licenseComments = stringCombiner(masterPackage.getLicenseComments(), tempPackage.getLicenseComments());
	                	 masterPackage.setLicenseComments(licenseComments);
	                	 
	                	 //process to merge summary
	                	 String summary = stringCombiner(masterPackage.getSummary(), tempPackage.getSummary());
	                	 masterPackage.setSummary(summary);
	                	 
	                	 //process to merge description
	                	 String description = stringCombiner(masterPackage.getDescription(), tempPackage.getDescription());
	                	 masterPackage.setDescription(description);
	                	 
	                	 //process to merge user defined columns
	                	 String usrDefined = stringCombiner(masterPackage.getComment(), tempPackage.getComment());
	                	 masterPackage.setComment(usrDefined);

	                	 retval.set(index, masterPackage);
	                }
	                
				}
				
			}
			return packagesResult;
		}
        
	    /** A method to check license from file information through mapLicenseInfo function. If it is a
	     * non-standard license, check this particular license in the Mapper class. Replace the current 
	     * license value with the return value from the mapNonStdLicInMap method. After all, all the declared
	     * license should remain the same and non-standard license should replaced with updated id
	     * @param doc
	     * @param temp
	     * @return licFromFile
	     */
		public AnyLicenseInfo[] checkLicenseFromFile(SpdxDocument doc, SpdxPackage temp){
			AnyLicenseInfo[] licFromFile = temp.getLicenseInfoFromFiles();
			for(int k = 0; k < licFromFile.length; k++){
				if(licFromFile[k] instanceof ExtractedLicenseInfo){
					AnyLicenseInfo tempLicense = mapper.mapNonStdLicInMap(doc, licFromFile[k]);
					licFromFile[k] = tempLicense;
				}
			}			
			return licFromFile;
		}
		
		/** A method to clone the packages in the list 
		 * @param packagesArray
		 * @return clonedPackagesArray
		 */
		public List clonePackages(List packagesList){
			List clonedPackagesList = Lists.newArrayList();
			for(int h = 0; h < packagesList.size(); h++){
				clonedPackagesList.add(packagesList.get(h).clone());
			}
			return clonedPackagesList;
		}
		
		/**
		 * A method to combine two string variables into one string
		 * @param line1
		 * @param line2
		 * @return
		 */
		public String stringCombiner(String line1, String line2){
			StringBuilder buffer = new StringBuilder();
			if(line1.isEmpty() && line2.isEmpty()){
				return buffer.toString();
			}
			else if(!line1.isEmpty()){
				buffer.append(line1);
				if(!line1.equalsIgnoreCase(line2) && !line2.isEmpty()){
					buffer.append(" " + line2);
				}
			}
			else{
				buffer.append(line2);
			}
			return buffer.toString();
		}
		
		/**
		 * A method to combine two licenses into one
		 * @param lic1
		 * @param lic2
		 * @return
		 * @throws InvalidLicenseStringException
		 */
		public AnyLicenseInfo licsCombiner(AnyLicenseInfo lic1, AnyLicenseInfo lic2) 
				throws InvalidLicenseStringException{
			StringBuilder buffer = new StringBuilder(lic1.toString());
			if(!lic1.equals(lic2)){
				buffer.append(" " + lic2.toString());
			}
			return LicenseInfoFactory.parseSPDXLicenseString(buffer.toString());
		}
		
		/**
		 * A method to collect all skipped files from input SPDX package.
		 * @return excludedFileNamesArray
		 * @throws InvalidSPDXAnalysisException
		 */
		public String[] collectSkippedFiles(SpdxPackage main, SpdxPackage sub ) throws InvalidSPDXAnalysisException{
			List excludedFileNamesList = Lists.newArrayList();
			String[] retval = sub.getPackageVerificationCode().getExcludedFileNames();
			String[] skippedFileInMain = main.getPackageVerificationCode().getExcludedFileNames();
			
			if(skippedFileInMain.length == 0 && retval.length == 0){
				String[] excludedFileNamesArray = new String[0];
				return excludedFileNamesArray;
			}
			else if(skippedFileInMain.length != 0){
				excludedFileNamesList = Lists.newArrayList(skippedFileInMain);
			}
			else if(excludedFileNamesList.isEmpty() && retval.length != 0){
				excludedFileNamesList = Lists.newArrayList(retval);
			}
			else{
					for(int k = 0; k < retval.length; k++){
						boolean foundNameMatch = false;
						for(int q = 0; q < excludedFileNamesList.size(); q++){
							if(retval[k].equalsIgnoreCase(excludedFileNamesList.get(q))){
								foundNameMatch = true;
							}
						}
						if(!foundNameMatch){
							excludedFileNamesList.add(retval[k]);
						}
					}
				}
			String[] excludedFileNamesArray = new String[excludedFileNamesList.size()];
			excludedFileNamesList.toArray(excludedFileNamesArray);
			excludedFileNamesList.clear();
			return excludedFileNamesArray;
		}

		@Deprecated
		/**
		 * 
		 * @param subDocs
		 * @return
		 * @throws InvalidSPDXAnalysisException
		 */
		public String translateSubDelcaredLicsIntoComments(SpdxDocument[] subDocs) throws InvalidSPDXAnalysisException{
			SpdxLicenseMapper mapper = new SpdxLicenseMapper();
				if(!mapper.isNonStdLicIdMapEmpty()){
					StringBuilder buffer = new StringBuilder(packageInfoResult.getLicenseComments() 
							+ ". This package merged several packages and the sub-package contain the following licenses:");

					for(int k = 0; k < subDocs.length; k++){
						if(mapper.docInNonStdLicIdMap(subDocs[k])){
						 List tempList = subDocs[k].getDocumentContainer().findAllPackages();
						  for(int h = 0; h < tempList.size(); h++){
							AnyLicenseInfo license = tempList.get(h).getLicenseDeclared();
							AnyLicenseInfo result = mapper.mapLicenseInfo(subDocs[k], license); 
							buffer.append(tempList.get(h).getPackageFileName());
							buffer.append(" (" + result.toString() + ") ");
						  }
						}else{
							List tempList = subDocs[k].getDocumentContainer().findAllPackages();
							for(int q = 0; q < tempList.size(); q++){
							buffer.append(tempList.get(q).getPackageFileName());
							buffer.append(" (" + tempList.get(q).getLicenseDeclared().toString() + ") ");
						  }
						}
					}			
					return buffer.toString();
			}else{
				return packageInfoResult.getLicenseComments();
			}
		}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy