diff --git a/gradle.properties b/gradle.properties index 2d5ef26a1..b980da60f 100644 --- a/gradle.properties +++ b/gradle.properties @@ -3,7 +3,7 @@ # https://www.jetbrains.com/intellij-repository/snapshots # ideaVersion=2020.1 -#ideaVersion=2020.2.1 +#ideaVersion=2020.2 # pluginGroup=de.plushnikov.intellij.plugin pluginName=lombok-plugin diff --git a/src/main/java/de/plushnikov/intellij/plugin/processor/SynchronizedProcessor.java b/src/main/java/de/plushnikov/intellij/plugin/processor/SynchronizedProcessor.java index ab25a0d5c..9d7f14153 100644 --- a/src/main/java/de/plushnikov/intellij/plugin/processor/SynchronizedProcessor.java +++ b/src/main/java/de/plushnikov/intellij/plugin/processor/SynchronizedProcessor.java @@ -40,11 +40,11 @@ public Collection verifyAnnotation(@NotNull PsiAnnotation psiAnno if (null != psiMethod) { if (psiMethod.hasModifierProperty(PsiModifier.ABSTRACT)) { problemNewBuilder.addError(LombokBundle.message("inspection.message.synchronized.legal.only.on.concrete.methods"), - PsiQuickFixFactory.createModifierListFix(psiMethod, PsiModifier.ABSTRACT, false, false) + PsiQuickFixFactory.createModifierListFix(psiMethod, PsiModifier.ABSTRACT, false, false) ); } - @NlsSafe final String lockFieldName = PsiAnnotationUtil.getStringAnnotationValue(psiAnnotation, "value", ""); + final String lockFieldName = PsiAnnotationUtil.getStringAnnotationValue(psiAnnotation, "value", ""); if (StringUtil.isNotEmpty(lockFieldName)) { final PsiClass containingClass = psiMethod.getContainingClass(); diff --git a/src/main/java/de/plushnikov/intellij/plugin/provider/LombokAugmentProvider.java b/src/main/java/de/plushnikov/intellij/plugin/provider/LombokAugmentProvider.java index 2d486a95c..ee651d314 100644 --- a/src/main/java/de/plushnikov/intellij/plugin/provider/LombokAugmentProvider.java +++ b/src/main/java/de/plushnikov/intellij/plugin/provider/LombokAugmentProvider.java @@ -2,6 +2,8 @@ import com.intellij.openapi.application.ApplicationManager; import com.intellij.openapi.diagnostic.Logger; +import com.intellij.openapi.util.RecursionGuard; +import com.intellij.openapi.util.RecursionManager; import com.intellij.psi.*; import com.intellij.psi.augment.PsiAugmentProvider; import com.intellij.psi.impl.source.PsiExtensibleClass; @@ -71,7 +73,7 @@ public List getAugments(@NotNull PsiElement elemen } @NotNull - @Override +// @Override public List getAugments(@NotNull PsiElement element, @NotNull final Class type, @Nullable String nameHint) { @@ -90,14 +92,63 @@ public List getAugments(@NotNull PsiElement elemen return emptyResult; } - // All invoker of AugmentProvider already make caching - // and we want to try to skip recursive calls completely + final LombokValueProvider result; + if (type == PsiField.class) { + result = new FieldLombokProvider<>(type, psiClass, nameHint); + } else if (type == PsiMethod.class) { + result = new MethodLombokProvider<>(type, psiClass, nameHint); + } else { + result = new ClassLombokProvider<>(type, psiClass, nameHint); + } -/// final String message = String.format("Process call for type: %s class: %s", type.getSimpleName(), psiClass.getQualifiedName()); -// log.info(">>>" + message); - final List result = getPsis(psiClass, type, nameHint); -// log.info("<<<" + message); - return result; + final List computed = result.compute(); + return null != computed ? computed : emptyResult; + } + + private static class FieldLombokProvider extends LombokValueProvider { + private static final RecursionGuard ourGuard = RecursionManager.createGuard("lombok.augment.field"); + + FieldLombokProvider(Class type, PsiClass psiClass, String nameHint) { + super(type, psiClass, ourGuard, nameHint); + } + } + + private static class MethodLombokProvider extends LombokValueProvider { + private static final RecursionGuard ourGuard = RecursionManager.createGuard("lombok.augment.method"); + + MethodLombokProvider(Class type, PsiClass psiClass, String nameHint) { + super(type, psiClass, ourGuard, nameHint); + } + } + + private static class ClassLombokProvider extends LombokValueProvider { + private static final RecursionGuard ourGuard = RecursionManager.createGuard("lombok.augment.class"); + + ClassLombokProvider(Class type, PsiClass psiClass, String nameHint) { + super(type, psiClass, ourGuard, nameHint); + } + } + + private abstract static class LombokValueProvider { + private final Class type; + private final PsiClass psiClass; + private final RecursionGuard recursionGuard; + private final String nameHint; + + LombokValueProvider(Class type, PsiClass psiClass, RecursionGuard recursionGuard, String nameHint) { + this.type = type; + this.psiClass = psiClass; + this.recursionGuard = recursionGuard; + this.nameHint = nameHint; + } + + public List compute() { + return recursionGuard.doPreventingRecursion(psiClass, true, this::computeIntern); + } + + private List computeIntern() { + return getPsis(psiClass, type, nameHint); + } } @NotNull diff --git a/src/main/java/de/plushnikov/intellij/plugin/settings/ProjectSettingsPage.java b/src/main/java/de/plushnikov/intellij/plugin/settings/ProjectSettingsPage.java index 33ae480a9..5fdcc166d 100644 --- a/src/main/java/de/plushnikov/intellij/plugin/settings/ProjectSettingsPage.java +++ b/src/main/java/de/plushnikov/intellij/plugin/settings/ProjectSettingsPage.java @@ -41,7 +41,6 @@ public JComponent createComponent() { private void initFromSettings() { myEnableExtensionMethodSupport.setSelected(ProjectSettings.isEnabled(myProject, ProjectSettings.IS_EXTENSION_METHOD_ENABLED, false)); - myEnableLombokVersionWarning.setSelected(ProjectSettings.isEnabled(myProject, ProjectSettings.IS_LOMBOK_VERSION_CHECK_ENABLED, false)); } @@ -59,8 +58,6 @@ public void apply() { ProjectSettings.setEnabled(myProject, ProjectSettings.IS_LOMBOK_VERSION_CHECK_ENABLED, myEnableLombokVersionWarning.isSelected()); // Redo code checking and highlighting. - LombokAugmentProvider.onConfigChange(); - PsiManager.getInstance(myProject).dropPsiCaches(); DaemonCodeAnalyzer.getInstance(myProject).restart(); } diff --git a/src/main/java/de/plushnikov/intellij/plugin/util/LombokProcessorUtil.java b/src/main/java/de/plushnikov/intellij/plugin/util/LombokProcessorUtil.java index f25adb5d3..1a46c8007 100644 --- a/src/main/java/de/plushnikov/intellij/plugin/util/LombokProcessorUtil.java +++ b/src/main/java/de/plushnikov/intellij/plugin/util/LombokProcessorUtil.java @@ -25,14 +25,14 @@ public final class LombokProcessorUtil { @NonNls private static final String ACCESS_LEVEL_NONE = "NONE"; - private static final Map ACCESS_LEVEL_MAP = new HashMap<>() {{ + private static final Map ACCESS_LEVEL_MAP = new HashMap() {{ put(PsiUtil.ACCESS_LEVEL_PUBLIC, ACCESS_LEVEL_PUBLIC); put(PsiUtil.ACCESS_LEVEL_PACKAGE_LOCAL, ACCESS_LEVEL_PACKAGE_LOCAL); put(PsiUtil.ACCESS_LEVEL_PROTECTED, ACCESS_LEVEL_PROTECTED); put(PsiUtil.ACCESS_LEVEL_PRIVATE, ACCESS_LEVEL_PRIVATE); }}; - private static final Map VALUE_ACCESS_LEVEL_MAP = new HashMap<>() {{ + private static final Map VALUE_ACCESS_LEVEL_MAP = new HashMap() {{ put(ACCESS_LEVEL_PUBLIC, PsiModifier.PUBLIC); put(ACCESS_LEVEL_PACKAGE_LOCAL, PsiModifier.PACKAGE_LOCAL); put(ACCESS_LEVEL_PROTECTED, PsiModifier.PROTECTED); diff --git a/src/main/java/de/plushnikov/intellij/plugin/util/PsiAnnotationSearchUtil.java b/src/main/java/de/plushnikov/intellij/plugin/util/PsiAnnotationSearchUtil.java index 0817877f2..8a4fdd5e1 100644 --- a/src/main/java/de/plushnikov/intellij/plugin/util/PsiAnnotationSearchUtil.java +++ b/src/main/java/de/plushnikov/intellij/plugin/util/PsiAnnotationSearchUtil.java @@ -1,32 +1,107 @@ package de.plushnikov.intellij.plugin.util; import com.intellij.openapi.util.text.StringUtil; -import com.intellij.psi.PsiAnnotation; -import com.intellij.psi.PsiJavaCodeReferenceElement; -import com.intellij.psi.PsiModifierList; -import com.intellij.psi.PsiModifierListOwner; +import com.intellij.psi.*; +import com.intellij.psi.impl.source.SourceJavaCodeReference; +import com.intellij.util.ArrayUtil; import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import java.util.Collection; -import java.util.Objects; import java.util.regex.Pattern; -import java.util.stream.Stream; -public class PsiAnnotationSearchUtil { +public final class PsiAnnotationSearchUtil { @Nullable public static PsiAnnotation findAnnotation(@NotNull PsiModifierListOwner psiModifierListOwner, @NotNull String annotationFQN) { - return psiModifierListOwner.getAnnotation(annotationFQN); + return findAnnotationQuick(psiModifierListOwner.getModifierList(), annotationFQN); } @Nullable public static PsiAnnotation findAnnotation(@NotNull PsiModifierListOwner psiModifierListOwner, @NotNull String... annotationFQNs) { - return Stream.of(annotationFQNs).map(psiModifierListOwner::getAnnotation).filter(Objects::nonNull).findAny().orElse(null); + return findAnnotationQuick(psiModifierListOwner.getModifierList(), annotationFQNs); } - public static boolean isAnnotatedWith(@NotNull PsiModifierListOwner psiModifierListOwner, @NotNull String annotationFQN) { - return psiModifierListOwner.hasAnnotation(annotationFQN); + @Nullable + private static PsiAnnotation findAnnotationQuick(@Nullable PsiAnnotationOwner annotationOwner, @NotNull String qualifiedName) { + if (annotationOwner == null) { + return null; + } + + PsiAnnotation[] annotations = annotationOwner.getAnnotations(); + if (annotations.length == 0) { + return null; + } + + final String shortName = StringUtil.getShortName(qualifiedName); + + for (PsiAnnotation annotation : annotations) { + PsiJavaCodeReferenceElement referenceElement = annotation.getNameReferenceElement(); + if (null != referenceElement) { + final String referenceName = referenceElement.getReferenceName(); + if (shortName.equals(referenceName)) { + + if (referenceElement.isQualified() && referenceElement instanceof SourceJavaCodeReference) { + String possibleFullQualifiedName = ((SourceJavaCodeReference) referenceElement).getClassNameText(); + if (qualifiedName.equals(possibleFullQualifiedName)) { + return annotation; + } + } + + final String annotationQualifiedName = annotation.getQualifiedName(); + if (null != annotationQualifiedName && qualifiedName.endsWith(annotationQualifiedName)) { + return annotation; + } + } + } + } + + return null; + } + + @Nullable + private static PsiAnnotation findAnnotationQuick(@Nullable PsiAnnotationOwner annotationOwner, @NotNull String... qualifiedNames) { + if (annotationOwner == null || qualifiedNames.length == 0) { + return null; + } + + PsiAnnotation[] annotations = annotationOwner.getAnnotations(); + if (annotations.length == 0) { + return null; + } + + final String[] shortNames = new String[qualifiedNames.length]; + for (int i = 0; i < qualifiedNames.length; i++) { + shortNames[i] = StringUtil.getShortName(qualifiedNames[i]); + } + + for (PsiAnnotation annotation : annotations) { + final PsiJavaCodeReferenceElement referenceElement = annotation.getNameReferenceElement(); + if (null != referenceElement) { + final String referenceName = referenceElement.getReferenceName(); + if (ArrayUtil.find(shortNames, referenceName) > -1) { + + if (referenceElement.isQualified() && referenceElement instanceof SourceJavaCodeReference) { + final String possibleFullQualifiedName = ((SourceJavaCodeReference) referenceElement).getClassNameText(); + + if (ArrayUtil.find(qualifiedNames, possibleFullQualifiedName) > -1) { + return annotation; + } + } + + final String annotationQualifiedName = annotation.getQualifiedName(); + if (ArrayUtil.find(qualifiedNames, annotationQualifiedName) > -1) { + return annotation; + } + } + } + } + + return null; + } + + public static boolean isAnnotatedWith(@NotNull PsiModifierListOwner psiModifierListOwner, @NotNull String annotationTypeName) { + return null != findAnnotation(psiModifierListOwner, annotationTypeName); } public static boolean isNotAnnotatedWith(@NotNull PsiModifierListOwner psiModifierListOwner, String annotationTypeName) { diff --git a/src/test/java/de/plushnikov/intellij/plugin/AbstractLombokLightCodeInsightTestCase.java b/src/test/java/de/plushnikov/intellij/plugin/AbstractLombokLightCodeInsightTestCase.java index 8c05c8205..342aca30a 100644 --- a/src/test/java/de/plushnikov/intellij/plugin/AbstractLombokLightCodeInsightTestCase.java +++ b/src/test/java/de/plushnikov/intellij/plugin/AbstractLombokLightCodeInsightTestCase.java @@ -1,5 +1,6 @@ package de.plushnikov.intellij.plugin; +import com.intellij.openapi.util.RecursionManager; import com.intellij.openapi.util.io.FileUtil; import com.intellij.openapi.util.text.StringUtil; import com.intellij.openapi.vfs.VirtualFile; @@ -35,6 +36,9 @@ public void setUp() throws Exception { super.setUp(); loadLombokLibrary(); + + //TODO disable assertions for the moment + RecursionManager.disableMissedCacheAssertions(myFixture.getProjectDisposable()); } protected void loadLombokLibrary() { diff --git a/src/test/java/de/plushnikov/intellij/plugin/inspection/LombokInspectionTest.java b/src/test/java/de/plushnikov/intellij/plugin/inspection/LombokInspectionTest.java index 12b8cc2ea..2a59b3a44 100644 --- a/src/test/java/de/plushnikov/intellij/plugin/inspection/LombokInspectionTest.java +++ b/src/test/java/de/plushnikov/intellij/plugin/inspection/LombokInspectionTest.java @@ -1,5 +1,6 @@ package de.plushnikov.intellij.plugin.inspection; +import com.intellij.openapi.util.RecursionManager; import com.intellij.openapi.util.registry.Registry; import com.intellij.testFramework.LightProjectDescriptor; import com.siyeh.ig.LightJavaInspectionTestCase; @@ -16,6 +17,9 @@ public void setUp() throws Exception { LombokTestUtil.loadLombokLibrary(myFixture.getProjectDisposable(), getModule()); Registry.get("platform.random.idempotence.check.rate").setValue(1, getTestRootDisposable()); + + //TODO disable assertions for the moment + RecursionManager.disableMissedCacheAssertions(myFixture.getProjectDisposable()); } @NotNull diff --git a/test-manual/src/main/java/de/plushnikov/builder/singularperformance/SingularPerformanceIssueDemo.java b/test-manual/src/main/java/de/plushnikov/builder/singularperformance/SingularPerformanceIssueDemo.java index 286a554fe..2c095d665 100644 --- a/test-manual/src/main/java/de/plushnikov/builder/singularperformance/SingularPerformanceIssueDemo.java +++ b/test-manual/src/main/java/de/plushnikov/builder/singularperformance/SingularPerformanceIssueDemo.java @@ -9,7 +9,26 @@ public class SingularPerformanceIssueDemo { @Singular private Collection number_0_strings; - + @Singular + private Collection number_1_strings; + @Singular + private Collection number_2_strings; + @Singular + private Collection number_3_strings; + @Singular + private Collection number_4_strings; + @Singular + private Collection number_5_strings; + @Singular + private Collection number_6_strings; + @Singular + private Collection number_7_strings; + @Singular + private Collection number_8_strings; + @Singular + private Collection number_9_strings; + @Singular + private Collection number_10_strings; public static void main(String[] args) { for (int i = 0; i < 20; i++) {