From 41fdf6b7ab000409076bd52162fb5094811d1a15 Mon Sep 17 00:00:00 2001 From: Daniel Espendiller Date: Sat, 6 Apr 2024 10:36:48 +0200 Subject: [PATCH] rearrange public property completion / inspection code --- ...tePropertyServiceInjectionContributor.java | 548 ------------------ .../completion/ServicePropertyInsertUtil.java | 271 +++++++++ .../PhpPropertyArgumentIntention.java | 6 +- src/main/resources/META-INF/plugin.xml | 5 - ...opertyServiceInjectionContributorTest.java | 398 ------------- .../ServicePropertyInsertUtilTest.java | 126 ++++ ...utor.php => ServicePropertyInsertUtil.php} | 0 7 files changed, 400 insertions(+), 954 deletions(-) delete mode 100644 src/main/java/fr/adrienbrault/idea/symfony2plugin/completion/IncompletePropertyServiceInjectionContributor.java create mode 100644 src/main/java/fr/adrienbrault/idea/symfony2plugin/completion/ServicePropertyInsertUtil.java delete mode 100644 src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/IncompletePropertyServiceInjectionContributorTest.java create mode 100644 src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/ServicePropertyInsertUtilTest.java rename src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/fixtures/{IncompletePropertyServiceInjectionContributor.php => ServicePropertyInsertUtil.php} (100%) diff --git a/src/main/java/fr/adrienbrault/idea/symfony2plugin/completion/IncompletePropertyServiceInjectionContributor.java b/src/main/java/fr/adrienbrault/idea/symfony2plugin/completion/IncompletePropertyServiceInjectionContributor.java deleted file mode 100644 index 944a382d6..000000000 --- a/src/main/java/fr/adrienbrault/idea/symfony2plugin/completion/IncompletePropertyServiceInjectionContributor.java +++ /dev/null @@ -1,548 +0,0 @@ -package fr.adrienbrault.idea.symfony2plugin.completion; - -import com.intellij.application.options.CodeStyle; -import com.intellij.codeInsight.completion.*; -import com.intellij.codeInsight.lookup.LookupElement; -import com.intellij.codeInsight.lookup.LookupElementBuilder; -import com.intellij.openapi.project.Project; -import com.intellij.patterns.PatternCondition; -import com.intellij.patterns.PlatformPatterns; -import com.intellij.patterns.PsiElementPattern; -import com.intellij.patterns.StandardPatterns; -import com.intellij.psi.PsiElement; -import com.intellij.psi.SmartPsiElementPointer; -import com.intellij.psi.util.PsiTreeUtil; -import com.intellij.util.ProcessingContext; -import com.jetbrains.php.PhpIndex; -import com.jetbrains.php.PhpPresentationUtil; -import com.jetbrains.php.config.PhpLanguageFeature; -import com.jetbrains.php.config.PhpLanguageLevel; -import com.jetbrains.php.lang.formatter.PhpCodeStyleSettings; -import com.jetbrains.php.lang.lexer.PhpTokenTypes; -import com.jetbrains.php.lang.parser.PhpElementTypes; -import com.jetbrains.php.lang.psi.PhpPsiElementFactory; -import com.jetbrains.php.lang.psi.PhpPsiUtil; -import com.jetbrains.php.lang.psi.elements.*; -import com.jetbrains.php.lang.psi.elements.impl.PhpPromotedFieldParameterImpl; -import com.jetbrains.php.lang.psi.resolve.types.PhpType; -import com.jetbrains.php.refactoring.PhpNameStyle; -import com.jetbrains.php.refactoring.PhpNameUtil; -import com.jetbrains.php.refactoring.PhpRefactoringUtil; -import com.jetbrains.php.refactoring.changeSignature.PhpChangeSignatureProcessor; -import com.jetbrains.php.refactoring.changeSignature.PhpParameterInfo; -import com.jetbrains.php.refactoring.introduce.introduceField.PhpIntroduceFieldHandler; -import fr.adrienbrault.idea.symfony2plugin.Symfony2Icons; -import fr.adrienbrault.idea.symfony2plugin.Symfony2ProjectComponent; -import fr.adrienbrault.idea.symfony2plugin.util.PhpElementsUtil; -import fr.adrienbrault.idea.symfony2plugin.util.dict.ServiceUtil; -import kotlin.Triple; -import org.apache.commons.lang3.StringUtils; -import org.jetbrains.annotations.NotNull; -import org.jetbrains.annotations.Nullable; - -import java.util.*; -import java.util.stream.Collectors; - -/** - * @author Daniel Espendiller - */ -public class IncompletePropertyServiceInjectionContributor extends CompletionContributor { - private static final String[] CLASS_TYPE_NAMES = {"interface", "abstract", "decorator"}; - - private static final PatternCondition THIS_FIELD_NAME_PATTERN = new PatternCondition<>("this pattern") { - @Override - public boolean accepts(@NotNull FieldReference fieldReference, ProcessingContext context) { - PhpExpression classReference = fieldReference.getClassReference(); - if (classReference instanceof Variable && "this".equals(classReference.getName())) { - return true; - } - - return false; - } - }; - - @NotNull - private PsiElementPattern.Capture getThisFieldNamePattern() { - return PlatformPatterns.psiElement().withElementType(PhpTokenTypes.IDENTIFIER) - .withParent( - PlatformPatterns.psiElement(FieldReference.class).with(THIS_FIELD_NAME_PATTERN) - ); - } - - public IncompletePropertyServiceInjectionContributor() { - extend(CompletionType.BASIC, getThisFieldNamePattern(), new CompletionProvider<>() { - @Override - protected void addCompletions(@NotNull CompletionParameters completionParameters, @NotNull ProcessingContext context, @NotNull CompletionResultSet result) { - if(!Symfony2ProjectComponent.isEnabled(completionParameters.getPosition())) { - return; - } - - result.restartCompletionOnPrefixChange(StandardPatterns.string().longerThan(1)); - - String completedText = getCompletedText(completionParameters); - if (completedText == null || completedText.length() < 2) { - return; - } - - PsiElement originalPosition = completionParameters.getOriginalPosition(); - PhpClass phpClassScope = PsiTreeUtil.getParentOfType(originalPosition, PhpClass.class); - if (phpClassScope == null || !ServiceUtil.isPhpClassAService(phpClassScope)) { - return; - } - - Method constructor = phpClassScope.getConstructor(); - if (constructor != null && constructor.getAccess() != PhpModifier.Access.PUBLIC) { - return; - } - - Set fields = new HashSet<>(); - Set types = new HashSet<>(); - - for (Field field : phpClassScope.getFields()) { - if (!field.isConstant()) { - fields.add(field.getName().toLowerCase()); - types.addAll(field.getType().getTypes() - .stream() - .flatMap((f) -> PhpIndex.getInstance(phpClassScope.getProject()).getAnyByFQN(f).stream()) - .distinct() - .map(PhpNamedElement::getFQN) - .map(String::toLowerCase) - .collect(Collectors.toList())); - } - } - - if (constructor != null) { - for (Parameter parameter : constructor.getParameters()) { - fields.add(parameter.getName().toLowerCase()); - - types.addAll(parameter.getType().getTypes() - .stream() - .flatMap((f) -> PhpIndex.getInstance(phpClassScope.getProject()).getAnyByFQN(f).stream()) - .distinct() - .map(PhpNamedElement::getFQN) - .map(String::toLowerCase) - .collect(Collectors.toList())); - } - } - - Collection> completed = new ArrayList<>(); - - for (Triple injection : getInjectionService(originalPosition.getProject())) { - for (String fqn : injection.getSecond()) { - if (types.contains(fqn.toLowerCase())) { - continue; - } - - String propertyName = injection.getFirst(); - if (propertyName == null) { - int i = fqn.lastIndexOf("\\"); - if (i > 0) { - propertyName = fqn.substring(i + 1); - } else { - propertyName = fqn; - } - - propertyName = StringUtils.removeEndIgnoreCase(propertyName, "interface"); - propertyName = StringUtils.removeEndIgnoreCase(propertyName, "abstract"); - propertyName = StringUtils.removeEndIgnoreCase(propertyName, "factory"); - - // propertyName = fr.adrienbrault.idea.symfony2plugin.util.StringUtils.camelize(propertyName); - propertyName = propertyName.substring(0, 1).toLowerCase() + propertyName.substring(1); - } - - if (StringUtils.isBlank(propertyName) || !propertyName.toLowerCase().startsWith(completedText.toLowerCase()) || fields.contains(propertyName.toLowerCase())) { - continue; - } - - Collection anyByFQN = PhpIndex.getInstance(originalPosition.getProject()).getAnyByFQN(fqn); - if (anyByFQN.isEmpty()) { - continue; - } - - LookupElementBuilder lookupElementBuilder = LookupElementBuilder.createWithSmartPointer(propertyName, phpClassScope) - .withInsertHandler(new LookupElementInsertHandler(propertyName, fqn)) - .withIcon(Symfony2Icons.SERVICE_OPACITY) - .withTypeText(StringUtils.stripStart(fqn, "\\"), true); - - result.addElement(lookupElementBuilder); - - completed.add(new Triple<>(propertyName, fqn, injection.getThird())); - - break; - } - } - - if (completed.size() <= 5) { - int maxMethodUntilUnsureCompletion = 5; - - if (completed.size() <= 2) { - maxMethodUntilUnsureCompletion = 15; - } else if(completed.size() <= 3) { - maxMethodUntilUnsureCompletion = 10; - } - - for (Triple injection : completed) { - String fqn = injection.getSecond(); - - Collection anyByFQN = PhpIndex.getInstance(originalPosition.getProject()).getAnyByFQN(fqn); - if (anyByFQN.isEmpty()) { - continue; - } - - String propertyName = injection.getFirst(); - - PhpClass next = anyByFQN.iterator().next(); - - Set methods = next.getMethods() - .stream() - .filter(method -> !method.isStatic() && method.getAccess() == PhpModifier.Access.PUBLIC && !method.isDeprecated()) - .collect(Collectors.toSet()); - - // whitelisted - Set whitelisted = methods.stream() - .filter(method -> Arrays.stream(injection.getThird()).anyMatch(s -> method.getName().equalsIgnoreCase(s))) - .collect(Collectors.toSet()); - - Set whitelistedMethodNames = new HashSet<>(); - for (Method method : whitelisted) { - appendMethod(result, phpClassScope, fqn, propertyName, method); - whitelistedMethodNames.add(method.getName().toLowerCase()); - } - - if (methods.size() <= maxMethodUntilUnsureCompletion) { - for (Method method : methods) { - if (whitelistedMethodNames.contains(method.getName())) { - continue; - } - - appendMethod(result, phpClassScope, fqn, propertyName, method); - } - } - } - } - } - - private void appendMethod(@NotNull CompletionResultSet result, PhpClass parentOfType, String fqn, String propertyName, Method method) { - LookupElementBuilder lookupElementBuilder = LookupElementBuilder.createWithSmartPointer(propertyName + "->" + method.getName() + "();", parentOfType) - .withPresentableText(propertyName + "->" + method.getName()) - .withInsertHandler(new LookupElementInsertHandler(propertyName, fqn)) - .withIcon(Symfony2Icons.SERVICE_OPACITY) - .withTypeText(StringUtils.stripStart(fqn, "\\"), true); - - Parameter[] parameters = method.getParameters(); - if (parameters.length > 0) { - lookupElementBuilder = lookupElementBuilder.withTailText(PhpPresentationUtil.formatParameters(null, method.getParameters()).toString(), true); - } - - result.addElement(lookupElementBuilder); - } - }); - } - - private static Collection> getInjectionService(@NotNull Project project) { - // @TODO: fill this list based on project usage - return new ArrayList<>() {{ - add(new Triple<>(null, new String[] {"\\Symfony\\Contracts\\Translation\\TranslatorInterface", "Symfony\\Component\\Translation\\TranslatorInterface"}, new String[0])); - add(new Triple<>(null, new String[] {"\\Symfony\\Component\\HttpFoundation\\RequestStack"}, new String[0])); - add(new Triple<>(null, new String[] {"\\Twig\\Environment"}, new String[] {"render"})); - add(new Triple<>("twig", new String[] {"\\Twig\\Environment"}, new String[] {"render"})); - add(new Triple<>(null, new String[] {"\\Psr\\Log\\LoggerInterface"}, new String[] {"error", "debug", "info"})); - add(new Triple<>(null, new String[] {"\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"}, new String[] {"generate"})); - add(new Triple<>("router", new String[] {"\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"}, new String[] {"generate"})); - add(new Triple<>(null, new String[] {"\\Doctrine\\ORM\\EntityManagerInterface", "\\Doctrine\\Persistence\\ObjectManager"}, new String[] {"flush", "find", "remove", "persist", "getRepository"})); - }}; - } - - public static List getInjectionService(@NotNull Project project, @NotNull String propertyNameFind) { - return getInjectionService(project, propertyNameFind, null); - } - - public static List getInjectionService(@NotNull Project project, @NotNull String propertyNameFindRaw, @Nullable String methodName) { - // @TODO: fill this list based on project usage - - final Set propertyNameFind = new HashSet<>(); - propertyNameFind.add(normalizeClassTypeKeywords(propertyNameFindRaw)); - - // LoggerInterface $fooBarLogger - if (propertyNameFindRaw.endsWith("Logger") && !propertyNameFindRaw.equalsIgnoreCase("logger")) { - propertyNameFind.add("logger"); - } - - Map servicesMatch = new HashMap<>(); - - HashMap alias = new HashMap<>() {{ - put("twig", "\\Twig\\Environment"); - put("template", "\\Twig\\Environment"); - put("router", "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - put("em", "Doctrine\\ORM\\EntityManagerInterface"); - put("om", "\\Doctrine\\Persistence\\ObjectManager"); - }}; - - for (String property : propertyNameFind) { - if (alias.containsKey(property.toLowerCase())) { - String key = property.toLowerCase(); - if (!PhpIndex.getInstance(project).getAnyByFQN(alias.get(key)).isEmpty()) { - String fqn = alias.get(key); - servicesMatch.put(fqn, new Match(fqn, 4)); - } - } - } - - // try to find partial ending match for normalized properties: fooBarCar => barCar - String classPropertyNameForEndingMatch = fr.adrienbrault.idea.symfony2plugin.util.StringUtils.underscore(StringUtils.strip(propertyNameFindRaw, "_")); - for (String replace : CLASS_TYPE_NAMES) { - classPropertyNameForEndingMatch = StringUtils.removeEndIgnoreCase(classPropertyNameForEndingMatch, replace); - classPropertyNameForEndingMatch = StringUtils.removeStartIgnoreCase(classPropertyNameForEndingMatch, replace); - } - - classPropertyNameForEndingMatch = fr.adrienbrault.idea.symfony2plugin.util.StringUtils.camelize(classPropertyNameForEndingMatch, true); - - // collect partial match with least 3 parts - Set endingMatches = new HashSet<>(); - List nameParts = PhpNameUtil.splitName(classPropertyNameForEndingMatch); - if (nameParts.size() > 2) { - PhpCodeStyleSettings settings = CodeStyle.getCustomSettings(PhpPsiElementFactory.createPsiFileFromText(project, " fr.adrienbrault.idea.symfony2plugin.util.StringUtils.underscore(s).split("_").length > 2) - .collect(Collectors.toSet()) - ); - } - - HashSet objects = new HashSet<>(); - - objects.addAll(PhpIndex.getInstance(project).getAllClassFqns(PrefixMatcher.ALWAYS_TRUE)); - objects.addAll(PhpIndex.getInstance(project).getAllInterfacesFqns(PrefixMatcher.ALWAYS_TRUE)); - - Set collect = objects.stream().filter(s -> { - int i = s.lastIndexOf("\\"); - if (i > 0) { - if (s.toLowerCase().contains("\\test\\")) { - return false; - } - - s = s.substring(i); - } - - return !s.endsWith("Test") - && !s.toLowerCase().contains("_phpstan_") - && !s.toLowerCase().contains("rectorprefix"); - }).collect(Collectors.toSet()); - - for (String fqn : collect) { - // Bar\Foo => Foo - int i = fqn.lastIndexOf("\\"); - String classPropertyNameRaw = i > 0 - ? fqn.substring(i + 1) - : fqn; - - String classPropertyName = normalizeClassTypeKeywords(classPropertyNameRaw); - if (StringUtils.isBlank(classPropertyName)) { - continue; - } - - int weight; - if (propertyNameFind.stream().anyMatch(classPropertyName::equalsIgnoreCase)) { - // direct property match - weight = 3; - } else if(endingMatches.stream().anyMatch(s -> classPropertyName.toLowerCase().endsWith(s.toLowerCase()))) { - // partial property with ending match - weight = 1; - } else { - continue; - } - - Collection anyByFQN = PhpIndex.getInstance(project).getAnyByFQN(fqn); - if (anyByFQN.isEmpty()) { - continue; - } - - if (methodName != null && !hasMethodMatch(methodName, anyByFQN)) { - weight -= 4; - } - - if (anyByFQN.stream().anyMatch(PhpClass::isInterface)) { - weight += 2; - - // Symfony\Contracts\EventDispatcher\EventDispatcherInterface - // Psr\Log\LoggerInterface - if (fqn.toLowerCase().contains("\\contracts\\") && fqn.toLowerCase().contains("\\symfony\\")) { - weight += 2; - } else if(fqn.toLowerCase().contains("\\psr\\")) { - weight += 3; - } - } - - if (anyByFQN.stream().anyMatch(PhpClass::isAbstract)) { - weight += 1; - } - - if (classPropertyNameRaw.toLowerCase().contains("decorator")) { - weight -= 3; - } - - if (servicesMatch.containsKey(fqn)) { - servicesMatch.get(fqn).modifyWeight(weight); - } else { - servicesMatch.put(fqn, new Match(fqn, weight)); - } - } - - return servicesMatch.values().stream() - .sorted((o1, o2) -> Integer.compare(o2.weight, o1.weight)) - .map(m -> m.fqn) - .collect(Collectors.toList()); - } - - private static class Match { - private final String fqn; - private int weight = 0; - - public Match(@NotNull String fqn, int weight) { - this.fqn = fqn; - this.modifyWeight(weight); - } - - public void modifyWeight(int weight) { - this.weight += weight; - } - } - - private static boolean hasMethodMatch(@NotNull String methodName, Collection anyByFQN) { - return anyByFQN.stream() - .anyMatch(phpClass -> phpClass.findMethodByName(methodName) != null); - } - - private static String normalizeClassTypeKeywords(@NotNull String classPropertyName) { - classPropertyName = classPropertyName.replaceAll("_", "").toLowerCase(); - - for (String replace : CLASS_TYPE_NAMES) { - classPropertyName = StringUtils.removeEndIgnoreCase(classPropertyName, replace); - classPropertyName = StringUtils.removeStartIgnoreCase(classPropertyName, replace); - } - - return classPropertyName; - } - - @Nullable - private String getCompletedText(@NotNull CompletionParameters completionParameters) { - PsiElement originalPosition = completionParameters.getOriginalPosition(); - if (originalPosition != null) { - String text = originalPosition.getText(); - if (!text.isEmpty()) { - return text; - } - } - - PsiElement position = completionParameters.getPosition(); - String text = position.getText().toLowerCase().replace("intellijidearulezzz", ""); - if (!text.isEmpty()) { - return text; - } - - return null; - } - - private record LookupElementInsertHandler(String propertyName, String typePhpClass) implements InsertHandler { - @Override - public void handleInsert(@NotNull InsertionContext context, @NotNull LookupElement item) { - SmartPsiElementPointer parentOfType2 = (SmartPsiElementPointer) item.getObject(); - PhpClass parentOfType = parentOfType2.getElement(); - if (parentOfType == null) { - return; - } - - Method constructor = PhpIntroduceFieldHandler.getOrCreateConstructor(parentOfType); - if (constructor == null) { - return; - } - - // use + constructor(Foo $foo) - String importedClass = PhpElementsUtil.insertUseIfNecessary(parentOfType, typePhpClass); - PhpParameterInfo phpParameterInfo = new PhpParameterInfo(0, propertyName); - phpParameterInfo.setType(new PhpType().add(typePhpClass), importedClass); - - // find added parameter; should mmostly the last - Collection parameterInfos = List.of(phpParameterInfo); - PhpChangeSignatureProcessor.addParameterToFunctionSignature(parentOfType.getProject(), constructor, parameterInfos); - Parameter parameter = Arrays.stream(constructor.getParameters()) - .filter(parameter1 -> propertyName.equalsIgnoreCase(parameter1.getName())) - .findFirst() - .orElse(null); - - // add $this->foo - // readonly, constructor property promotion currently not supported; or handled automatically by code - if (parameter != null) { - PhpRefactoringUtil.initializeFieldsByParameters(parentOfType, List.of(parameter), PhpModifier.Access.PRIVATE); - } - - // move caret inside the function - String lookupString = item.getLookupString(); - if (lookupString.endsWith(");")) { - context.getEditor().getCaretModel().moveCaretRelatively(-2, 0, false, false, true); - } - } - } - - public static void appendPropertyInjection(@NotNull PhpClass phpClass, @NotNull String propertyName, @NotNull String typePhpClass) { - Method constructor = PhpIntroduceFieldHandler.getOrCreateConstructor(phpClass); - if (constructor == null) { - return; - } - - // use + constructor(Foo $foo) - String importedClass = PhpElementsUtil.insertUseIfNecessary(phpClass, typePhpClass); - - // "private readonly Foo $foo" - if (shouldUsePropertyPromotion(constructor)) { - String readonlyProperty = !phpClass.isReadonly() ? "readonly " : ""; - - Parameter parameter = PhpPsiElementFactory.createComplexParameter(phpClass.getProject(), String.format("private %s%s $%s", readonlyProperty, importedClass, propertyName)); - Parameter parameterToInsertAfter = PhpChangeSignatureProcessor.findParameterToInsertAfter(constructor); - if (parameterToInsertAfter != null) { - addParameterAfter(constructor, parameter, parameterToInsertAfter); - } else if (constructor.getParameters().length == 0) { - PhpChangeSignatureProcessor.appendParameterToParameterList(constructor, parameter); - } - - return; - } - - PhpParameterInfo phpParameterInfo = new PhpParameterInfo(0, propertyName); - phpParameterInfo.setType(new PhpType().add(typePhpClass), importedClass); - - // find added parameter; should mostly the last - PhpChangeSignatureProcessor.addParameterToFunctionSignature(phpClass.getProject(), constructor, List.of(phpParameterInfo)); - - Parameter parameter = Arrays.stream(constructor.getParameters()) - .filter(parameter1 -> propertyName.equalsIgnoreCase(parameter1.getName())) - .findFirst() - .orElse(null); - - // add $this->foo - if (parameter != null) { - PhpRefactoringUtil.initializeFieldsByParameters(phpClass, List.of(parameter), PhpModifier.Access.PRIVATE); - } - } - - private static @Nullable Parameter addParameterAfter(@NotNull Function function, @NotNull Parameter parameter, @NotNull Parameter parameterToInsertAfter) { - PsiElement parameterList = PhpPsiUtil.getChildOfType(function, PhpElementTypes.PARAMETER_LIST); - assert parameterList != null; - return (Parameter)parameterList.addAfter(parameter, parameterList.addAfter(PhpPsiElementFactory.createComma(parameterList.getProject()), parameterToInsertAfter)); - } - - public static boolean shouldUsePropertyPromotion(@NotNull Function function) { - Parameter[] parameters = function.getParameters(); - if (parameters.length == 0) { - return PhpLanguageLevel.current(function.getProject()).hasFeature(PhpLanguageFeature.PROPERTY_PROMOTION); - } - - for (Parameter parameter : parameters) { - if (parameter instanceof PhpPromotedFieldParameterImpl) { - return true; - } - } - - return false; - } -} diff --git a/src/main/java/fr/adrienbrault/idea/symfony2plugin/completion/ServicePropertyInsertUtil.java b/src/main/java/fr/adrienbrault/idea/symfony2plugin/completion/ServicePropertyInsertUtil.java new file mode 100644 index 000000000..bdad60956 --- /dev/null +++ b/src/main/java/fr/adrienbrault/idea/symfony2plugin/completion/ServicePropertyInsertUtil.java @@ -0,0 +1,271 @@ +package fr.adrienbrault.idea.symfony2plugin.completion; + +import com.intellij.application.options.CodeStyle; +import com.intellij.codeInsight.completion.PrefixMatcher; +import com.intellij.openapi.project.Project; +import com.intellij.psi.PsiElement; +import com.jetbrains.php.PhpIndex; +import com.jetbrains.php.config.PhpLanguageFeature; +import com.jetbrains.php.config.PhpLanguageLevel; +import com.jetbrains.php.lang.formatter.PhpCodeStyleSettings; +import com.jetbrains.php.lang.parser.PhpElementTypes; +import com.jetbrains.php.lang.psi.PhpPsiElementFactory; +import com.jetbrains.php.lang.psi.PhpPsiUtil; +import com.jetbrains.php.lang.psi.elements.*; +import com.jetbrains.php.lang.psi.elements.impl.PhpPromotedFieldParameterImpl; +import com.jetbrains.php.lang.psi.resolve.types.PhpType; +import com.jetbrains.php.refactoring.PhpNameStyle; +import com.jetbrains.php.refactoring.PhpNameUtil; +import com.jetbrains.php.refactoring.PhpRefactoringUtil; +import com.jetbrains.php.refactoring.changeSignature.PhpChangeSignatureProcessor; +import com.jetbrains.php.refactoring.changeSignature.PhpParameterInfo; +import com.jetbrains.php.refactoring.introduce.introduceField.PhpIntroduceFieldHandler; +import fr.adrienbrault.idea.symfony2plugin.util.PhpElementsUtil; +import org.apache.commons.lang3.StringUtils; +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; + +import java.util.*; +import java.util.stream.Collectors; + +/** + * @author Daniel Espendiller + */ +public class ServicePropertyInsertUtil { + private static final String[] CLASS_TYPE_NAMES = {"interface", "abstract", "decorator"}; + + public static List getInjectionService(@NotNull Project project, @NotNull String propertyNameFind) { + return getInjectionService(project, propertyNameFind, null); + } + + public static List getInjectionService(@NotNull Project project, @NotNull String propertyNameFindRaw, @Nullable String methodName) { + // @TODO: fill this list based on project usage + + final Set propertyNameFind = new HashSet<>(); + propertyNameFind.add(normalizeClassTypeKeywords(propertyNameFindRaw)); + + // LoggerInterface $fooBarLogger + if (propertyNameFindRaw.endsWith("Logger") && !propertyNameFindRaw.equalsIgnoreCase("logger")) { + propertyNameFind.add("logger"); + } + + Map servicesMatch = new HashMap<>(); + + HashMap alias = new HashMap<>() {{ + put("twig", "\\Twig\\Environment"); + put("template", "\\Twig\\Environment"); + put("router", "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + put("em", "Doctrine\\ORM\\EntityManagerInterface"); + put("om", "\\Doctrine\\Persistence\\ObjectManager"); + }}; + + for (String property : propertyNameFind) { + if (alias.containsKey(property.toLowerCase())) { + String key = property.toLowerCase(); + if (!PhpIndex.getInstance(project).getAnyByFQN(alias.get(key)).isEmpty()) { + String fqn = alias.get(key); + servicesMatch.put(fqn, new Match(fqn, 4)); + } + } + } + + // try to find partial ending match for normalized properties: fooBarCar => barCar + String classPropertyNameForEndingMatch = fr.adrienbrault.idea.symfony2plugin.util.StringUtils.underscore(StringUtils.strip(propertyNameFindRaw, "_")); + for (String replace : CLASS_TYPE_NAMES) { + classPropertyNameForEndingMatch = StringUtils.removeEndIgnoreCase(classPropertyNameForEndingMatch, replace); + classPropertyNameForEndingMatch = StringUtils.removeStartIgnoreCase(classPropertyNameForEndingMatch, replace); + } + + classPropertyNameForEndingMatch = fr.adrienbrault.idea.symfony2plugin.util.StringUtils.camelize(classPropertyNameForEndingMatch, true); + + // collect partial match with least 3 parts + Set endingMatches = new HashSet<>(); + List nameParts = PhpNameUtil.splitName(classPropertyNameForEndingMatch); + if (nameParts.size() > 2) { + PhpCodeStyleSettings settings = CodeStyle.getCustomSettings(PhpPsiElementFactory.createPsiFileFromText(project, " fr.adrienbrault.idea.symfony2plugin.util.StringUtils.underscore(s).split("_").length > 2) + .collect(Collectors.toSet()) + ); + } + + HashSet objects = new HashSet<>(); + + objects.addAll(PhpIndex.getInstance(project).getAllClassFqns(PrefixMatcher.ALWAYS_TRUE)); + objects.addAll(PhpIndex.getInstance(project).getAllInterfacesFqns(PrefixMatcher.ALWAYS_TRUE)); + + Set collect = objects.stream().filter(s -> { + int i = s.lastIndexOf("\\"); + if (i > 0) { + if (s.toLowerCase().contains("\\test\\")) { + return false; + } + + s = s.substring(i); + } + + return !s.endsWith("Test") + && !s.toLowerCase().contains("_phpstan_") + && !s.toLowerCase().contains("ecsprefix") + && !s.toLowerCase().contains("_humbugbox") + && !s.toLowerCase().contains("rectorprefix"); + }).collect(Collectors.toSet()); + + for (String fqn : collect) { + // Bar\Foo => Foo + int i = fqn.lastIndexOf("\\"); + String classPropertyNameRaw = i > 0 + ? fqn.substring(i + 1) + : fqn; + + String classPropertyName = normalizeClassTypeKeywords(classPropertyNameRaw); + if (StringUtils.isBlank(classPropertyName)) { + continue; + } + + int weight; + if (propertyNameFind.stream().anyMatch(classPropertyName::equalsIgnoreCase)) { + // direct property match + weight = 3; + } else if(endingMatches.stream().anyMatch(s -> classPropertyName.toLowerCase().endsWith(s.toLowerCase()))) { + // partial property with ending match + weight = 1; + } else { + continue; + } + + Collection anyByFQN = PhpIndex.getInstance(project).getAnyByFQN(fqn); + if (anyByFQN.isEmpty()) { + continue; + } + + if (methodName != null && !hasMethodMatch(methodName, anyByFQN)) { + weight -= 4; + } + + if (anyByFQN.stream().anyMatch(PhpClass::isInterface)) { + weight += 2; + + // Symfony\Contracts\EventDispatcher\EventDispatcherInterface + // Psr\Log\LoggerInterface + if (fqn.toLowerCase().contains("\\contracts\\") && fqn.toLowerCase().contains("\\symfony\\")) { + weight += 2; + } else if(fqn.toLowerCase().contains("\\psr\\")) { + weight += 3; + } + } + + if (anyByFQN.stream().anyMatch(PhpClass::isAbstract)) { + weight += 1; + } + + if (classPropertyNameRaw.toLowerCase().contains("decorator")) { + weight -= 3; + } + + if (servicesMatch.containsKey(fqn)) { + servicesMatch.get(fqn).modifyWeight(weight); + } else { + servicesMatch.put(fqn, new Match(fqn, weight)); + } + } + + return servicesMatch.values().stream() + .sorted((o1, o2) -> Integer.compare(o2.weight, o1.weight)) + .map(m -> m.fqn) + .collect(Collectors.toList()); + } + + private static class Match { + private final String fqn; + private int weight = 0; + + public Match(@NotNull String fqn, int weight) { + this.fqn = fqn; + this.modifyWeight(weight); + } + + public void modifyWeight(int weight) { + this.weight += weight; + } + } + + private static boolean hasMethodMatch(@NotNull String methodName, Collection anyByFQN) { + return anyByFQN.stream() + .anyMatch(phpClass -> phpClass.findMethodByName(methodName) != null); + } + + private static String normalizeClassTypeKeywords(@NotNull String classPropertyName) { + classPropertyName = classPropertyName.replaceAll("_", "").toLowerCase(); + + for (String replace : CLASS_TYPE_NAMES) { + classPropertyName = StringUtils.removeEndIgnoreCase(classPropertyName, replace); + classPropertyName = StringUtils.removeStartIgnoreCase(classPropertyName, replace); + } + + return classPropertyName; + } + + public static void appendPropertyInjection(@NotNull PhpClass phpClass, @NotNull String propertyName, @NotNull String typePhpClass) { + Method constructor = PhpIntroduceFieldHandler.getOrCreateConstructor(phpClass); + if (constructor == null) { + return; + } + + // use + constructor(Foo $foo) + String importedClass = PhpElementsUtil.insertUseIfNecessary(phpClass, typePhpClass); + + // "private readonly Foo $foo" + if (shouldUsePropertyPromotion(constructor)) { + String readonlyProperty = !phpClass.isReadonly() ? "readonly " : ""; + + Parameter parameter = PhpPsiElementFactory.createComplexParameter(phpClass.getProject(), String.format("private %s%s $%s", readonlyProperty, importedClass, propertyName)); + Parameter parameterToInsertAfter = PhpChangeSignatureProcessor.findParameterToInsertAfter(constructor); + if (parameterToInsertAfter != null) { + addParameterAfter(constructor, parameter, parameterToInsertAfter); + } else if (constructor.getParameters().length == 0) { + PhpChangeSignatureProcessor.appendParameterToParameterList(constructor, parameter); + } + + return; + } + + PhpParameterInfo phpParameterInfo = new PhpParameterInfo(0, propertyName); + phpParameterInfo.setType(new PhpType().add(typePhpClass), importedClass); + + // find added parameter; should mostly the last + PhpChangeSignatureProcessor.addParameterToFunctionSignature(phpClass.getProject(), constructor, List.of(phpParameterInfo)); + + Parameter parameter = Arrays.stream(constructor.getParameters()) + .filter(parameter1 -> propertyName.equalsIgnoreCase(parameter1.getName())) + .findFirst() + .orElse(null); + + // add $this->foo + if (parameter != null) { + PhpRefactoringUtil.initializeFieldsByParameters(phpClass, List.of(parameter), PhpModifier.Access.PRIVATE); + } + } + + private static void addParameterAfter(@NotNull Function function, @NotNull Parameter parameter, @NotNull Parameter parameterToInsertAfter) { + PsiElement parameterList = PhpPsiUtil.getChildOfType(function, PhpElementTypes.PARAMETER_LIST); + assert parameterList != null; + parameterList.addAfter(parameter, parameterList.addAfter(PhpPsiElementFactory.createComma(parameterList.getProject()), parameterToInsertAfter)); + } + + public static boolean shouldUsePropertyPromotion(@NotNull Function function) { + Parameter[] parameters = function.getParameters(); + if (parameters.length == 0) { + return PhpLanguageLevel.current(function.getProject()).hasFeature(PhpLanguageFeature.PROPERTY_PROMOTION); + } + + for (Parameter parameter : parameters) { + if (parameter instanceof PhpPromotedFieldParameterImpl) { + return true; + } + } + + return false; + } +} diff --git a/src/main/java/fr/adrienbrault/idea/symfony2plugin/dic/intention/PhpPropertyArgumentIntention.java b/src/main/java/fr/adrienbrault/idea/symfony2plugin/dic/intention/PhpPropertyArgumentIntention.java index 40b7f9f74..5db9d868f 100644 --- a/src/main/java/fr/adrienbrault/idea/symfony2plugin/dic/intention/PhpPropertyArgumentIntention.java +++ b/src/main/java/fr/adrienbrault/idea/symfony2plugin/dic/intention/PhpPropertyArgumentIntention.java @@ -20,7 +20,7 @@ import com.jetbrains.php.PhpIndex; import com.jetbrains.php.lang.findUsages.PhpGotoTargetRendererProvider; import com.jetbrains.php.lang.psi.elements.*; -import fr.adrienbrault.idea.symfony2plugin.completion.IncompletePropertyServiceInjectionContributor; +import fr.adrienbrault.idea.symfony2plugin.completion.ServicePropertyInsertUtil; import fr.adrienbrault.idea.symfony2plugin.util.PhpElementsUtil; import fr.adrienbrault.idea.symfony2plugin.util.dict.ServiceUtil; import icons.SymfonyIcons; @@ -104,7 +104,7 @@ public void applyFix(@NotNull Project project, PsiFile file, @Nullable Editor ed } } - List injectionService = IncompletePropertyServiceInjectionContributor.getInjectionService(project, name, methodName) + List injectionService = ServicePropertyInsertUtil.getInjectionService(project, name, methodName) .stream() .map(s -> StringUtils.stripStart(s, "\\")) .toList(); @@ -153,7 +153,7 @@ private static void buildProperty(@NotNull Project project, @NotNull FieldRefere WriteCommandAction.writeCommandAction(project) .withName("Symfony: Add Property Service") - .run((ThrowableRunnable) () -> IncompletePropertyServiceInjectionContributor.appendPropertyInjection(phpClassScope, fieldReference.getName(), classFqn)); + .run((ThrowableRunnable) () -> ServicePropertyInsertUtil.appendPropertyInjection(phpClassScope, fieldReference.getName(), classFqn)); }); } catch (Throwable ignored) { } diff --git a/src/main/resources/META-INF/plugin.xml b/src/main/resources/META-INF/plugin.xml index 80a727059..3ea8302a7 100644 --- a/src/main/resources/META-INF/plugin.xml +++ b/src/main/resources/META-INF/plugin.xml @@ -249,11 +249,6 @@ - - - diff --git a/src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/IncompletePropertyServiceInjectionContributorTest.java b/src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/IncompletePropertyServiceInjectionContributorTest.java deleted file mode 100644 index ce341ee5c..000000000 --- a/src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/IncompletePropertyServiceInjectionContributorTest.java +++ /dev/null @@ -1,398 +0,0 @@ -package fr.adrienbrault.idea.symfony2plugin.tests.completion; - -import com.jetbrains.php.lang.PhpFileType; -import com.jetbrains.php.lang.psi.PhpPsiElementFactory; -import com.jetbrains.php.lang.psi.elements.PhpClass; -import fr.adrienbrault.idea.symfony2plugin.completion.IncompletePropertyServiceInjectionContributor; -import fr.adrienbrault.idea.symfony2plugin.tests.SymfonyLightCodeInsightFixtureTestCase; - -import java.util.List; - -/** - * @author Daniel Espendiller - * @see fr.adrienbrault.idea.symfony2plugin.completion.IncompletePropertyServiceInjectionContributor - */ -public class IncompletePropertyServiceInjectionContributorTest extends SymfonyLightCodeInsightFixtureTestCase { - public void setUp() throws Exception { - super.setUp(); - myFixture.copyFileToProject("IncompletePropertyServiceInjectionContributor.php"); - } - - public String getTestDataPath() { - return "src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/fixtures"; - } - - public void testInjectionCompletionUnknownPropertyProvidesInjectionCompletion() { - if (true) { - return; - } - - assertCompletionContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator", "translator->trans();" - ); - - assertCompletionContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator", "translator->trans();" - ); - } - - public void testInjectionCompletionUnknownPropertyProvidesWithConstructor() { - if (true) { - return; - } - - assertCompletionResultEquals(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator = $translator;\n" + - " }\n" + - "\n" + - " public function testFoo()\n" + - " {\n" + - " $this->translator\n" + - " }\n" + - "}", - lookupElement -> lookupElement.getLookupString().equals("translator") - ); - } - - public void testInjectionCompletionUnknownPropertyWithoutConstructorCompletion() { - if (true) { - return; - } - - assertCompletionResultEquals(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator = $translator;\n" + - " }\n" + - "\n" + - " public function testFoo()\n" + - " {\n" + - " $this->translator\n" + - " }\n" + - "}", - lookupElement -> lookupElement.getLookupString().equals("translator") - ); - } - - public void testInjectionCompletionNotProvidedForPrivateConstructor() { - if (true) { - return; - } - - assertCompletionNotContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator" - ); - } - - public void testInjectionCompletionNotProvidedForInvalidConstructor() { - if (true) { - return; - } - - assertCompletionNotContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator" - ); - } - - public void testInjectionCompletionNotProvidedForAlreadyExistingTypPath() { - if (true) { - return; - } - - assertCompletionNotContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator->trans();" - ); - - assertCompletionNotContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator" - ); - - assertCompletionNotContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator" - ); - - assertCompletionNotContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator" - ); - } - - public void testInjectionCompletionNotProvidedForNonService() { - if (true) { - return; - } - - assertCompletionNotContains(PhpFileType.INSTANCE, "translator\n" + - " }\n" + - "}", - "translator" - ); - } - - public void testAppendPropertyInjection() { - PhpClass fromText = PhpPsiElementFactory.createFromText(getProject(), PhpClass.class, " classes1 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "router"); - assertContainsElements(classes1, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - - List classes2 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "urlgenerator"); - assertContainsElements(classes2, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - - List classes3 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "urlGenerator"); - assertContainsElements(classes3, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - - List classes4 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "_urlGenerator"); - assertContainsElements(classes4, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - - List classes5 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "__url_generator"); - assertContainsElements(classes5, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - - List classes6 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "_router"); - assertContainsElements(classes6, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - - List classes7 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "foobar"); - assertContainsElements(classes7, "\\App\\Service\\FoobarInterface"); - - List classes8 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "_routerInterface"); - assertContainsElements(classes8, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - - List classes9 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "foobarCar"); - assertContainsElements(classes9, "\\App\\Service\\InterfaceFoobarCar"); - - List classes10 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "foobarCarInterface"); - assertContainsElements(classes10, "\\App\\Service\\InterfaceFoobarCar"); - - List classes11 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "fooBarLogger"); - assertContainsElements(classes11, "\\Psr\\Log\\LoggerInterface"); - - List classes12 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "foobarLongClassNameServiceFactory"); - assertContainsElements(classes12, "\\App\\Service\\FoobarLongClassNameServiceFactory"); - - List classes13 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "longClassNameServiceFactory"); - assertContainsElements(classes13, "\\App\\Service\\FoobarLongClassNameServiceFactory"); - - List classes14 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "nameServiceFactory"); - assertContainsElements(classes14, "\\App\\Service\\FoobarLongClassNameServiceFactory"); - - List classes15 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "serviceFactory"); - assertFalse(classes15.contains("\\App\\Service\\FoobarLongClassNameServiceFactory")); - - List classes16 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "_name_Service__Factory"); - assertContainsElements(classes16, "\\App\\Service\\FoobarLongClassNameServiceFactory"); - } - - public void testInjectionServiceWithName() { - List classes1 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "urlGenerator", "foobarUnknown"); - assertContainsElements(classes1, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - - List classes2 = IncompletePropertyServiceInjectionContributor.getInjectionService(getProject(), "urlGenerator", "generate"); - assertContainsElements(classes2, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); - } -} diff --git a/src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/ServicePropertyInsertUtilTest.java b/src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/ServicePropertyInsertUtilTest.java new file mode 100644 index 000000000..fb805263d --- /dev/null +++ b/src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/ServicePropertyInsertUtilTest.java @@ -0,0 +1,126 @@ +package fr.adrienbrault.idea.symfony2plugin.tests.completion; + +import com.jetbrains.php.lang.psi.PhpPsiElementFactory; +import com.jetbrains.php.lang.psi.elements.PhpClass; +import fr.adrienbrault.idea.symfony2plugin.completion.ServicePropertyInsertUtil; +import fr.adrienbrault.idea.symfony2plugin.tests.SymfonyLightCodeInsightFixtureTestCase; + +import java.util.List; + +/** + * @author Daniel Espendiller + * @see ServicePropertyInsertUtil + */ +public class ServicePropertyInsertUtilTest extends SymfonyLightCodeInsightFixtureTestCase { + public void setUp() throws Exception { + super.setUp(); + myFixture.copyFileToProject("ServicePropertyInsertUtil.php"); + } + + public String getTestDataPath() { + return "src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/fixtures"; + } + + public void testAppendPropertyInjection() { + PhpClass fromText = PhpPsiElementFactory.createFromText(getProject(), PhpClass.class, " classes1 = ServicePropertyInsertUtil.getInjectionService(getProject(), "router"); + assertContainsElements(classes1, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + + List classes2 = ServicePropertyInsertUtil.getInjectionService(getProject(), "urlgenerator"); + assertContainsElements(classes2, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + + List classes3 = ServicePropertyInsertUtil.getInjectionService(getProject(), "urlGenerator"); + assertContainsElements(classes3, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + + List classes4 = ServicePropertyInsertUtil.getInjectionService(getProject(), "_urlGenerator"); + assertContainsElements(classes4, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + + List classes5 = ServicePropertyInsertUtil.getInjectionService(getProject(), "__url_generator"); + assertContainsElements(classes5, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + + List classes6 = ServicePropertyInsertUtil.getInjectionService(getProject(), "_router"); + assertContainsElements(classes6, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + + List classes7 = ServicePropertyInsertUtil.getInjectionService(getProject(), "foobar"); + assertContainsElements(classes7, "\\App\\Service\\FoobarInterface"); + + List classes8 = ServicePropertyInsertUtil.getInjectionService(getProject(), "_routerInterface"); + assertContainsElements(classes8, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + + List classes9 = ServicePropertyInsertUtil.getInjectionService(getProject(), "foobarCar"); + assertContainsElements(classes9, "\\App\\Service\\InterfaceFoobarCar"); + + List classes10 = ServicePropertyInsertUtil.getInjectionService(getProject(), "foobarCarInterface"); + assertContainsElements(classes10, "\\App\\Service\\InterfaceFoobarCar"); + + List classes11 = ServicePropertyInsertUtil.getInjectionService(getProject(), "fooBarLogger"); + assertContainsElements(classes11, "\\Psr\\Log\\LoggerInterface"); + + List classes12 = ServicePropertyInsertUtil.getInjectionService(getProject(), "foobarLongClassNameServiceFactory"); + assertContainsElements(classes12, "\\App\\Service\\FoobarLongClassNameServiceFactory"); + + List classes13 = ServicePropertyInsertUtil.getInjectionService(getProject(), "longClassNameServiceFactory"); + assertContainsElements(classes13, "\\App\\Service\\FoobarLongClassNameServiceFactory"); + + List classes14 = ServicePropertyInsertUtil.getInjectionService(getProject(), "nameServiceFactory"); + assertContainsElements(classes14, "\\App\\Service\\FoobarLongClassNameServiceFactory"); + + List classes15 = ServicePropertyInsertUtil.getInjectionService(getProject(), "serviceFactory"); + assertFalse(classes15.contains("\\App\\Service\\FoobarLongClassNameServiceFactory")); + + List classes16 = ServicePropertyInsertUtil.getInjectionService(getProject(), "_name_Service__Factory"); + assertContainsElements(classes16, "\\App\\Service\\FoobarLongClassNameServiceFactory"); + } + + public void testInjectionServiceWithName() { + List classes1 = ServicePropertyInsertUtil.getInjectionService(getProject(), "urlGenerator", "foobarUnknown"); + assertContainsElements(classes1, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + + List classes2 = ServicePropertyInsertUtil.getInjectionService(getProject(), "urlGenerator", "generate"); + assertContainsElements(classes2, "\\Symfony\\Component\\Routing\\Generator\\UrlGeneratorInterface"); + } +} diff --git a/src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/fixtures/IncompletePropertyServiceInjectionContributor.php b/src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/fixtures/ServicePropertyInsertUtil.php similarity index 100% rename from src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/fixtures/IncompletePropertyServiceInjectionContributor.php rename to src/test/java/fr/adrienbrault/idea/symfony2plugin/tests/completion/fixtures/ServicePropertyInsertUtil.php