diff options
3 files changed, 102 insertions, 71 deletions
diff --git a/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndex.java b/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndex.java index c625b4a4ae2..2efaae29c67 100644 --- a/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndex.java +++ b/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/FileIndex.java @@ -151,9 +151,8 @@ public class FileIndex implements BatchComponent { @CheckForNull private InputFile newInputFile(ModuleFileSystem fileSystem, File sourceDir, String type, File file, String path) { - // File extension must be kept case-sensitive String lang = languageRecognizer.of(file); - if (lang == null) { + if (lang==null) { return null; } diff --git a/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/LanguageRecognizer.java b/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/LanguageRecognizer.java index 2214f24df1e..6357ee744ea 100644 --- a/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/LanguageRecognizer.java +++ b/sonar-batch/src/main/java/org/sonar/batch/scan/filesystem/LanguageRecognizer.java @@ -19,63 +19,82 @@ */ package org.sonar.batch.scan.filesystem; -import com.google.common.collect.Maps; +import com.google.common.annotations.VisibleForTesting; +import com.google.common.collect.HashMultimap; +import com.google.common.collect.SetMultimap; import org.apache.commons.io.FilenameUtils; import org.apache.commons.lang.StringUtils; import org.picocontainer.Startable; +import org.slf4j.LoggerFactory; import org.sonar.api.BatchComponent; import org.sonar.api.resources.Language; +import org.sonar.api.resources.Project; import javax.annotation.CheckForNull; import java.io.File; -import java.util.Map; +import java.util.Set; /** * Detect language of source files. Simplistic, based on file extensions. */ public class LanguageRecognizer implements BatchComponent, Startable { - /** - * Lower-case extension -> language - */ - private Map<String, String> byExtensions = Maps.newHashMap(); + static final String NO_EXTENSION = "<UNSET>"; + private final Project project; private final Language[] languages; - public LanguageRecognizer(Language[] languages) { + /** + * Lower-case extension -> languages + */ + private SetMultimap<String, String> langsByExtension = HashMultimap.create(); + + public LanguageRecognizer(Project project, Language[] languages) { + this.project = project; this.languages = languages; } @Override public void start() { for (Language language : languages) { - for (String suffix : language.getFileSuffixes()) { - String extension = sanitizeExtension(suffix); + if (language.getFileSuffixes().length == 0) { + langsByExtension.put(NO_EXTENSION, language.getKey()); - String s = byExtensions.get(extension); - if (s != null && !StringUtils.equals(s, language.getKey())) { - throw new IllegalStateException(String.format( - "File extension '%s' is declared by two languages: %s and %s", extension, s, language.getKey() - )); + } else { + for (String suffix : language.getFileSuffixes()) { + String extension = sanitizeExtension(suffix); + langsByExtension.put(extension, language.getKey()); } - byExtensions.put(extension, language.getKey()); + } + } + + for (String extension : langsByExtension.keySet()) { + Set<String> langs = langsByExtension.get(extension); + if (langs.size() > 1) { + warnConflict(extension, langs); } } } + @VisibleForTesting + void warnConflict(String extension, Set<String> langs) { + LoggerFactory.getLogger(LanguageRecognizer.class).warn(String.format( + "File extension '%s' is declared by several plugins: %s", extension, StringUtils.join(langs, ", ") + )); + } + @Override public void stop() { // do nothing } - // TODO what about cobol files without extension ? @CheckForNull String of(File file) { - String extension = FilenameUtils.getExtension(file.getName()); - if (StringUtils.isNotBlank(extension)) { - return byExtensions.get(StringUtils.lowerCase(extension)); - } - return null; + // multi-language is not supported yet. Filter on project language + String extension = sanitizeExtension(FilenameUtils.getExtension(file.getName())); + extension = StringUtils.defaultIfBlank(extension, NO_EXTENSION); + Set<String> langs = langsByExtension.get(extension); + return langs.contains(project.getLanguageKey()) ? project.getLanguageKey() : null; } static String sanitizeExtension(String suffix) { diff --git a/sonar-batch/src/test/java/org/sonar/batch/scan/filesystem/LanguageRecognizerTest.java b/sonar-batch/src/test/java/org/sonar/batch/scan/filesystem/LanguageRecognizerTest.java index dbba4e17311..b329f3bfbb6 100644 --- a/sonar-batch/src/test/java/org/sonar/batch/scan/filesystem/LanguageRecognizerTest.java +++ b/sonar-batch/src/test/java/org/sonar/batch/scan/filesystem/LanguageRecognizerTest.java @@ -22,11 +22,14 @@ package org.sonar.batch.scan.filesystem; import org.junit.Rule; import org.junit.Test; import org.junit.rules.TemporaryFolder; -import org.sonar.api.resources.Java; +import org.mockito.ArgumentMatcher; import org.sonar.api.resources.Language; +import org.sonar.api.resources.Project; + +import java.util.Set; import static org.fest.assertions.Assertions.assertThat; -import static org.fest.assertions.Fail.fail; +import static org.mockito.Mockito.*; public class LanguageRecognizerTest { @@ -43,13 +46,18 @@ public class LanguageRecognizerTest { @Test public void search_by_file_extension() throws Exception { - Language[] languages = new Language[]{Java.INSTANCE, new Cobol()}; - LanguageRecognizer recognizer = new LanguageRecognizer(languages); + Language[] languages = new Language[]{new MockLanguage("java", "java", "jav"), new MockLanguage("cobol", "cbl", "cob")}; + LanguageRecognizer recognizer = new LanguageRecognizer(newProject("java"), languages); recognizer.start(); - assertThat(recognizer.of(temp.newFile("Foo.java"))).isEqualTo(Java.KEY); - assertThat(recognizer.of(temp.newFile("abc.cbl"))).isEqualTo("cobol"); - assertThat(recognizer.of(temp.newFile("abc.CBL"))).isEqualTo("cobol"); + assertThat(recognizer.of(temp.newFile("Foo.java"))).isEqualTo("java"); + assertThat(recognizer.of(temp.newFile("Foo.JAVA"))).isEqualTo("java"); + assertThat(recognizer.of(temp.newFile("Foo.jav"))).isEqualTo("java"); + assertThat(recognizer.of(temp.newFile("Foo.Jav"))).isEqualTo("java"); + + // multi-language is not supported yet -> filter on project language + assertThat(recognizer.of(temp.newFile("abc.cbl"))).isNull(); + assertThat(recognizer.of(temp.newFile("abc.CBL"))).isNull(); assertThat(recognizer.of(temp.newFile("abc.php"))).isNull(); assertThat(recognizer.of(temp.newFile("abc"))).isNull(); recognizer.stop(); @@ -57,72 +65,77 @@ public class LanguageRecognizerTest { @Test public void fail_if_conflict_of_file_extensions() throws Exception { - Language[] languages = new Language[]{Java.INSTANCE, new Language() { - @Override - public String getKey() { - return "java2"; - } + Language[] languages = new Language[]{new MockLanguage("java", "java"), new MockLanguage("java2", "java", "java2")}; - @Override - public String getName() { - return "Java2"; - } + LanguageRecognizer recognizer = spy(new LanguageRecognizer(newProject("java"), languages)); + recognizer.start(); + verify(recognizer).warnConflict(eq("java"), argThat(new ArgumentMatcher<Set<String>>() { @Override - public String[] getFileSuffixes() { - return new String[]{"java2", "java"}; + public boolean matches(Object o) { + Set<String> set = (Set<String>) o; + return set.contains("java") && set.contains("java2") && set.size() == 2; } - }}; - - try { - new LanguageRecognizer(languages).start(); - fail(); - } catch (IllegalStateException e) { - assertThat(e.getMessage()).isEqualTo("File extension 'java' is declared by two languages: java and java2"); - } + })); } + @Test public void plugin_can_declare_a_file_extension_twice_for_case_sensitivity() throws Exception { - Language[] languages = new Language[]{new Language() { - @Override - public String getKey() { - return "abap"; - } + Language[] languages = new Language[]{new MockLanguage("abap", "abap", "ABAP")}; - @Override - public String getName() { - return "ABAP"; - } + LanguageRecognizer recognizer = new LanguageRecognizer(newProject("abap"), languages); + recognizer.start(); + assertThat(recognizer.of(temp.newFile("abc.abap"))).isEqualTo("abap"); + } - @Override - public String[] getFileSuffixes() { - return new String[]{"abap", "ABAP"}; - } - }}; + @Test + public void language_with_no_extension() throws Exception { + // abap here is associated to files without extension + Language[] languages = new Language[]{new MockLanguage("java", "java"), new MockLanguage("abap")}; - LanguageRecognizer recognizer = new LanguageRecognizer(languages); + // files without extension are detected only on abap projects + LanguageRecognizer recognizer = new LanguageRecognizer(newProject("java"), languages); recognizer.start(); - assertThat(recognizer.of(temp.newFile("abc.abap"))).isEqualTo("abap"); + assertThat(recognizer.of(temp.newFile("abc"))).isNull(); + recognizer.stop(); + + recognizer = new LanguageRecognizer(newProject("abap"), languages); + recognizer.start(); + assertThat(recognizer.of(temp.newFile("abc"))).isEqualTo("abap"); + recognizer.stop(); } - static class Cobol implements Language { + private Project newProject(String language) { + Project project = mock(Project.class); + when(project.getLanguageKey()).thenReturn(language); + return project; + } + + + static class MockLanguage implements Language { + private final String key; + private final String[] extensions; + + MockLanguage(String key, String... extensions) { + this.key = key; + this.extensions = extensions; + } + @Override public String getKey() { - return "cobol"; + return key; } @Override public String getName() { - return "Cobol"; + return key; } @Override public String[] getFileSuffixes() { - return new String[]{"cbl", "cob"}; + return extensions; } } - - } |