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.

FileIndexer.java 14KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. /*
  2. * SonarQube
  3. * Copyright (C) 2009-2019 SonarSource SA
  4. * mailto:info AT sonarsource DOT com
  5. *
  6. * This program is free software; you can redistribute it and/or
  7. * modify it under the terms of the GNU Lesser General Public
  8. * License as published by the Free Software Foundation; either
  9. * version 3 of the License, or (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  14. * Lesser General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU Lesser General Public License
  17. * along with this program; if not, write to the Free Software Foundation,
  18. * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  19. */
  20. package org.sonar.scanner.scan.filesystem;
  21. import java.io.IOException;
  22. import java.nio.file.LinkOption;
  23. import java.nio.file.Path;
  24. import java.util.Arrays;
  25. import java.util.function.BooleanSupplier;
  26. import javax.annotation.Nullable;
  27. import org.apache.commons.io.FilenameUtils;
  28. import org.sonar.api.CoreProperties;
  29. import org.sonar.api.batch.fs.InputFile;
  30. import org.sonar.api.batch.fs.InputFile.Type;
  31. import org.sonar.api.batch.fs.InputFileFilter;
  32. import org.sonar.api.batch.fs.internal.SensorStrategy;
  33. import org.sonar.api.batch.scm.IgnoreCommand;
  34. import org.sonar.api.notifications.AnalysisWarnings;
  35. import org.sonar.api.utils.MessageException;
  36. import org.sonar.api.utils.log.Logger;
  37. import org.sonar.api.utils.log.Loggers;
  38. import org.sonar.api.batch.fs.internal.DefaultIndexedFile;
  39. import org.sonar.api.batch.fs.internal.DefaultInputFile;
  40. import org.sonar.api.batch.fs.internal.DefaultInputModule;
  41. import org.sonar.api.batch.fs.internal.DefaultInputProject;
  42. import org.sonar.scanner.issue.ignore.scanner.IssueExclusionsLoader;
  43. import org.sonar.scanner.scan.ScanProperties;
  44. import org.sonar.scanner.util.ProgressReport;
  45. /**
  46. * Index input files into {@link InputComponentStore}.
  47. */
  48. public class FileIndexer {
  49. private static final Logger LOG = Loggers.get(FileIndexer.class);
  50. private final AnalysisWarnings analysisWarnings;
  51. private final ScanProperties properties;
  52. private final InputFileFilter[] filters;
  53. private final ProjectExclusionFilters projectExclusionFilters;
  54. private final ProjectCoverageAndDuplicationExclusions projectCoverageAndDuplicationExclusions;
  55. private final IssueExclusionsLoader issueExclusionsLoader;
  56. private final MetadataGenerator metadataGenerator;
  57. private final DefaultInputProject project;
  58. private final ScannerComponentIdGenerator scannerComponentIdGenerator;
  59. private final InputComponentStore componentStore;
  60. private final SensorStrategy sensorStrategy;
  61. private final LanguageDetection langDetection;
  62. private boolean warnInclusionsAlreadyLogged;
  63. private boolean warnExclusionsAlreadyLogged;
  64. private boolean warnCoverageExclusionsAlreadyLogged;
  65. private boolean warnDuplicationExclusionsAlreadyLogged;
  66. public FileIndexer(DefaultInputProject project, ScannerComponentIdGenerator scannerComponentIdGenerator, InputComponentStore componentStore,
  67. ProjectExclusionFilters projectExclusionFilters, ProjectCoverageAndDuplicationExclusions projectCoverageAndDuplicationExclusions, IssueExclusionsLoader issueExclusionsLoader,
  68. MetadataGenerator metadataGenerator, SensorStrategy sensorStrategy, LanguageDetection languageDetection, AnalysisWarnings analysisWarnings, ScanProperties properties,
  69. InputFileFilter[] filters) {
  70. this.project = project;
  71. this.scannerComponentIdGenerator = scannerComponentIdGenerator;
  72. this.componentStore = componentStore;
  73. this.projectCoverageAndDuplicationExclusions = projectCoverageAndDuplicationExclusions;
  74. this.issueExclusionsLoader = issueExclusionsLoader;
  75. this.metadataGenerator = metadataGenerator;
  76. this.sensorStrategy = sensorStrategy;
  77. this.langDetection = languageDetection;
  78. this.analysisWarnings = analysisWarnings;
  79. this.properties = properties;
  80. this.filters = filters;
  81. this.projectExclusionFilters = projectExclusionFilters;
  82. }
  83. public FileIndexer(DefaultInputProject project, ScannerComponentIdGenerator scannerComponentIdGenerator, InputComponentStore componentStore,
  84. ProjectExclusionFilters projectExclusionFilters, ProjectCoverageAndDuplicationExclusions projectCoverageAndDuplicationExclusions, IssueExclusionsLoader issueExclusionsLoader,
  85. MetadataGenerator metadataGenerator, SensorStrategy sensorStrategy, LanguageDetection languageDetection, AnalysisWarnings analysisWarnings, ScanProperties properties) {
  86. this(project, scannerComponentIdGenerator, componentStore, projectExclusionFilters, projectCoverageAndDuplicationExclusions, issueExclusionsLoader, metadataGenerator,
  87. sensorStrategy, languageDetection, analysisWarnings, properties, new InputFileFilter[0]);
  88. }
  89. void indexFile(DefaultInputModule module, ModuleExclusionFilters moduleExclusionFilters, ModuleCoverageAndDuplicationExclusions moduleCoverageAndDuplicationExclusions,
  90. Path sourceFile, Type type, ProgressReport progressReport, ProjectFileIndexer.ExclusionCounter exclusionCounter, @Nullable IgnoreCommand ignoreCommand)
  91. throws IOException {
  92. // get case of real file without resolving link
  93. Path realAbsoluteFile = sourceFile.toRealPath(LinkOption.NOFOLLOW_LINKS).toAbsolutePath().normalize();
  94. if (!realAbsoluteFile.startsWith(project.getBaseDir())) {
  95. LOG.warn("File '{}' is ignored. It is not located in project basedir '{}'.", realAbsoluteFile.toAbsolutePath(), project.getBaseDir());
  96. return;
  97. }
  98. if (!realAbsoluteFile.startsWith(module.getBaseDir())) {
  99. LOG.warn("File '{}' is ignored. It is not located in module basedir '{}'.", realAbsoluteFile.toAbsolutePath(), module.getBaseDir());
  100. return;
  101. }
  102. Path projectRelativePath = project.getBaseDir().relativize(realAbsoluteFile);
  103. Path moduleRelativePath = module.getBaseDir().relativize(realAbsoluteFile);
  104. boolean included = evaluateInclusionsFilters(moduleExclusionFilters, realAbsoluteFile, projectRelativePath, moduleRelativePath, type);
  105. if (!included) {
  106. exclusionCounter.increaseByPatternsCount();
  107. return;
  108. }
  109. boolean excluded = evaluateExclusionsFilters(moduleExclusionFilters, realAbsoluteFile, projectRelativePath, moduleRelativePath, type);
  110. if (excluded) {
  111. exclusionCounter.increaseByPatternsCount();
  112. return;
  113. }
  114. String language = langDetection.language(realAbsoluteFile, projectRelativePath);
  115. if (ignoreCommand != null && ignoreCommand.isIgnored(realAbsoluteFile)) {
  116. LOG.debug("File '{}' is excluded by the scm ignore settings.", realAbsoluteFile);
  117. exclusionCounter.increaseByScmCount();
  118. return;
  119. }
  120. DefaultIndexedFile indexedFile = new DefaultIndexedFile(realAbsoluteFile, project.key(),
  121. projectRelativePath.toString(),
  122. moduleRelativePath.toString(),
  123. type, language, scannerComponentIdGenerator.getAsInt(), sensorStrategy);
  124. DefaultInputFile inputFile = new DefaultInputFile(indexedFile, f -> metadataGenerator.setMetadata(module.key(), f, module.getEncoding()));
  125. if (language != null) {
  126. inputFile.setPublished(true);
  127. }
  128. if (!accept(inputFile)) {
  129. return;
  130. }
  131. checkIfAlreadyIndexed(inputFile);
  132. componentStore.put(module.key(), inputFile);
  133. issueExclusionsLoader.addMulticriteriaPatterns(inputFile);
  134. LOG.debug("'{}' indexed {}with language '{}'", projectRelativePath, type == Type.TEST ? "as test " : "", inputFile.language());
  135. evaluateCoverageExclusions(moduleCoverageAndDuplicationExclusions, inputFile);
  136. evaluateDuplicationExclusions(moduleCoverageAndDuplicationExclusions, inputFile);
  137. if (properties.preloadFileMetadata()) {
  138. inputFile.checkMetadata();
  139. }
  140. int count = componentStore.inputFiles().size();
  141. progressReport.message(count + " " + pluralizeFiles(count) + " indexed... (last one was " + inputFile.getProjectRelativePath() + ")");
  142. }
  143. private boolean evaluateInclusionsFilters(ModuleExclusionFilters moduleExclusionFilters, Path realAbsoluteFile, Path projectRelativePath, Path moduleRelativePath,
  144. InputFile.Type type) {
  145. if (!Arrays.equals(moduleExclusionFilters.getInclusionsConfig(type), projectExclusionFilters.getInclusionsConfig(type))) {
  146. // Module specific configuration
  147. return moduleExclusionFilters.isIncluded(realAbsoluteFile, moduleRelativePath, type);
  148. }
  149. boolean includedByProjectConfiguration = projectExclusionFilters.isIncluded(realAbsoluteFile, projectRelativePath, type);
  150. if (includedByProjectConfiguration) {
  151. return true;
  152. } else if (moduleExclusionFilters.isIncluded(realAbsoluteFile, moduleRelativePath, type)) {
  153. warnOnce(
  154. type == Type.MAIN ? CoreProperties.PROJECT_INCLUSIONS_PROPERTY : CoreProperties.PROJECT_TEST_INCLUSIONS_PROPERTY,
  155. FilenameUtils.normalize(projectRelativePath.toString(), true), () -> warnInclusionsAlreadyLogged, () -> warnInclusionsAlreadyLogged = true);
  156. return true;
  157. }
  158. return false;
  159. }
  160. private boolean evaluateExclusionsFilters(ModuleExclusionFilters moduleExclusionFilters, Path realAbsoluteFile, Path projectRelativePath, Path moduleRelativePath,
  161. InputFile.Type type) {
  162. if (!Arrays.equals(moduleExclusionFilters.getExclusionsConfig(type), projectExclusionFilters.getExclusionsConfig(type))) {
  163. // Module specific configuration
  164. return moduleExclusionFilters.isExcluded(realAbsoluteFile, moduleRelativePath, type);
  165. }
  166. boolean includedByProjectConfiguration = projectExclusionFilters.isExcluded(realAbsoluteFile, projectRelativePath, type);
  167. if (includedByProjectConfiguration) {
  168. return true;
  169. } else if (moduleExclusionFilters.isExcluded(realAbsoluteFile, moduleRelativePath, type)) {
  170. warnOnce(
  171. type == Type.MAIN ? CoreProperties.PROJECT_EXCLUSIONS_PROPERTY : CoreProperties.PROJECT_TEST_EXCLUSIONS_PROPERTY,
  172. FilenameUtils.normalize(projectRelativePath.toString(), true), () -> warnExclusionsAlreadyLogged, () -> warnExclusionsAlreadyLogged = true);
  173. return true;
  174. }
  175. return false;
  176. }
  177. private void checkIfAlreadyIndexed(DefaultInputFile inputFile) {
  178. if (componentStore.inputFile(inputFile.getProjectRelativePath()) != null) {
  179. throw MessageException.of("File " + inputFile + " can't be indexed twice. Please check that inclusion/exclusion patterns produce "
  180. + "disjoint sets for main and test files");
  181. }
  182. }
  183. private void evaluateCoverageExclusions(ModuleCoverageAndDuplicationExclusions moduleCoverageAndDuplicationExclusions, DefaultInputFile inputFile) {
  184. boolean excludedForCoverage = isExcludedForCoverage(moduleCoverageAndDuplicationExclusions, inputFile);
  185. inputFile.setExcludedForCoverage(excludedForCoverage);
  186. if (excludedForCoverage) {
  187. LOG.debug("File {} excluded for coverage", inputFile);
  188. }
  189. }
  190. private boolean isExcludedForCoverage(ModuleCoverageAndDuplicationExclusions moduleCoverageAndDuplicationExclusions, DefaultInputFile inputFile) {
  191. if (!Arrays.equals(moduleCoverageAndDuplicationExclusions.getCoverageExclusionConfig(), projectCoverageAndDuplicationExclusions.getCoverageExclusionConfig())) {
  192. // Module specific configuration
  193. return moduleCoverageAndDuplicationExclusions.isExcludedForCoverage(inputFile);
  194. }
  195. boolean excludedByProjectConfiguration = projectCoverageAndDuplicationExclusions.isExcludedForCoverage(inputFile);
  196. if (excludedByProjectConfiguration) {
  197. return true;
  198. } else if (moduleCoverageAndDuplicationExclusions.isExcludedForCoverage(inputFile)) {
  199. warnOnce(CoreProperties.PROJECT_COVERAGE_EXCLUSIONS_PROPERTY, inputFile.getProjectRelativePath(), () -> warnCoverageExclusionsAlreadyLogged,
  200. () -> warnCoverageExclusionsAlreadyLogged = true);
  201. return true;
  202. }
  203. return false;
  204. }
  205. private void evaluateDuplicationExclusions(ModuleCoverageAndDuplicationExclusions moduleCoverageAndDuplicationExclusions, DefaultInputFile inputFile) {
  206. boolean excludedForDuplications = isExcludedForDuplications(moduleCoverageAndDuplicationExclusions, inputFile);
  207. inputFile.setExcludedForDuplication(excludedForDuplications);
  208. if (excludedForDuplications) {
  209. LOG.debug("File {} excluded for duplication", inputFile);
  210. }
  211. }
  212. private boolean isExcludedForDuplications(ModuleCoverageAndDuplicationExclusions moduleCoverageAndDuplicationExclusions, DefaultInputFile inputFile) {
  213. if (!Arrays.equals(moduleCoverageAndDuplicationExclusions.getDuplicationExclusionConfig(), projectCoverageAndDuplicationExclusions.getDuplicationExclusionConfig())) {
  214. // Module specific configuration
  215. return moduleCoverageAndDuplicationExclusions.isExcludedForDuplication(inputFile);
  216. }
  217. boolean excludedByProjectConfiguration = projectCoverageAndDuplicationExclusions.isExcludedForDuplication(inputFile);
  218. if (excludedByProjectConfiguration) {
  219. return true;
  220. } else if (moduleCoverageAndDuplicationExclusions.isExcludedForDuplication(inputFile)) {
  221. warnOnce(CoreProperties.CPD_EXCLUSIONS, inputFile.getProjectRelativePath(), () -> warnDuplicationExclusionsAlreadyLogged,
  222. () -> warnDuplicationExclusionsAlreadyLogged = true);
  223. return true;
  224. }
  225. return false;
  226. }
  227. private void warnOnce(String propKey, String filePath, BooleanSupplier alreadyLoggedGetter, Runnable markAsLogged) {
  228. if (!alreadyLoggedGetter.getAsBoolean()) {
  229. String msg = "Specifying module-relative paths at project level in the property '" + propKey + "' is deprecated. " +
  230. "To continue matching files like '" + filePath + "', update this property so that patterns refer to project-relative paths.";
  231. LOG.warn(msg);
  232. analysisWarnings.addUnique(msg);
  233. markAsLogged.run();
  234. }
  235. }
  236. private boolean accept(InputFile indexedFile) {
  237. // InputFileFilter extensions. Might trigger generation of metadata
  238. for (InputFileFilter filter : filters) {
  239. if (!filter.accept(indexedFile)) {
  240. LOG.debug("'{}' excluded by {}", indexedFile, filter.getClass().getName());
  241. return false;
  242. }
  243. }
  244. return true;
  245. }
  246. private static String pluralizeFiles(int count) {
  247. return count == 1 ? "file" : "files";
  248. }
  249. }