blob: c27c77800c10f3601a01c2ab495b1f2ecaffcf4d [file] [log] [blame]
/*
* Copyright 2000-2014 JetBrains s.r.o.
*
* 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.jetbrains.idea.devkit.inspections;
import com.intellij.codeInspection.BaseJavaLocalInspectionTool;
import com.intellij.openapi.actionSystem.ActionGroup;
import com.intellij.openapi.actionSystem.AnAction;
import com.intellij.openapi.module.Module;
import com.intellij.openapi.module.ModuleUtil;
import com.intellij.openapi.paths.PathReference;
import com.intellij.openapi.project.Project;
import com.intellij.openapi.vfs.VirtualFile;
import com.intellij.psi.*;
import com.intellij.psi.search.GlobalSearchScope;
import com.intellij.psi.xml.*;
import com.intellij.util.containers.ContainerUtil;
import com.intellij.util.xml.DomFileElement;
import com.intellij.util.xml.DomUtil;
import com.intellij.util.xml.GenericAttributeValue;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.jetbrains.idea.devkit.DevKitBundle;
import org.jetbrains.idea.devkit.dom.Dependency;
import org.jetbrains.idea.devkit.dom.IdeaPlugin;
import org.jetbrains.idea.devkit.module.PluginModuleType;
import org.jetbrains.idea.devkit.util.ActionType;
import org.jetbrains.idea.devkit.util.ComponentType;
import org.jetbrains.idea.devkit.util.DescriptorUtil;
import java.util.List;
import java.util.Set;
/**
* @author swr
*/
public abstract class DevKitInspectionBase extends BaseJavaLocalInspectionTool {
@NotNull
public String getGroupDisplayName() {
return DevKitBundle.message("inspections.group.name");
}
@Nullable
protected static Set<PsiClass> getRegistrationTypes(PsiClass psiClass, boolean includeActions) {
final Project project = psiClass.getProject();
final PsiFile psiFile = psiClass.getContainingFile();
assert psiFile != null;
final VirtualFile virtualFile = psiFile.getVirtualFile();
if (virtualFile == null) return null;
final Module module = ModuleUtil.findModuleForFile(virtualFile, project);
if (module == null) return null;
if (PluginModuleType.isOfType(module)) {
return checkModule(module, psiClass, null, includeActions);
}
else {
Set<PsiClass> types = null;
final List<Module> modules = PluginModuleType.getCandidateModules(module);
for (Module m : modules) {
types = checkModule(m, psiClass, types, includeActions);
}
return types;
}
}
@Nullable
private static Set<PsiClass> checkModule(Module module, PsiClass psiClass, @Nullable Set<PsiClass> types, boolean includeActions) {
final XmlFile pluginXml = PluginModuleType.getPluginXml(module);
if (!DescriptorUtil.isPluginXml(pluginXml)) return types;
assert pluginXml != null;
final String qualifiedName = psiClass.getQualifiedName();
if (qualifiedName != null) {
final RegistrationTypeFinder finder = new RegistrationTypeFinder(psiClass, types);
// "main" plugin.xml
processPluginXml(pluginXml, finder, includeActions);
// <depends> plugin.xml files
final DomFileElement<IdeaPlugin> fileElement = DescriptorUtil.getIdeaPlugin(pluginXml);
for (Dependency dependency : fileElement.getRootElement().getDependencies()) {
final GenericAttributeValue<PathReference> configFileAttribute = dependency.getConfigFile();
if (!DomUtil.hasXml(configFileAttribute)) continue;
final PathReference configFile = configFileAttribute.getValue();
if (configFile != null) {
final PsiElement resolve = configFile.resolve();
if (!(resolve instanceof XmlFile)) continue;
final XmlFile depPluginXml = (XmlFile)resolve;
if (DescriptorUtil.isPluginXml(depPluginXml)) {
processPluginXml(depPluginXml, finder, includeActions);
}
}
}
types = finder.getTypes();
}
return types;
}
private static void processPluginXml(XmlFile xmlFile, RegistrationTypeFinder finder, boolean includeActions) {
final XmlDocument document = xmlFile.getDocument();
if (document == null) return;
final XmlTag rootTag = document.getRootTag();
if (rootTag == null) return;
DescriptorUtil.processComponents(rootTag, finder);
if (includeActions) {
DescriptorUtil.processActions(rootTag, finder);
}
}
@Nullable
protected static PsiElement getAttValueToken(@NotNull XmlAttribute attribute) {
final XmlAttributeValue valueElement = attribute.getValueElement();
if (valueElement == null) return null;
final PsiElement[] children = valueElement.getChildren();
if (children.length == 3 && children[1] instanceof XmlToken) {
return children[1];
}
if (children.length == 1 && children[0] instanceof PsiErrorElement) return null;
return valueElement;
}
protected static boolean isAbstract(PsiModifierListOwner checkedClass) {
return checkedClass.hasModifierProperty(PsiModifier.ABSTRACT);
}
protected static boolean isPublic(PsiModifierListOwner checkedClass) {
return checkedClass.hasModifierProperty(PsiModifier.PUBLIC);
}
protected static boolean isActionRegistered(PsiClass psiClass) {
final Set<PsiClass> registrationTypes = getRegistrationTypes(psiClass, true);
if (registrationTypes != null) {
for (PsiClass type : registrationTypes) {
if (AnAction.class.getName().equals(type.getQualifiedName())) return true;
if (ActionGroup.class.getName().equals(type.getQualifiedName())) return true;
}
}
return false;
}
private static class RegistrationTypeFinder implements ComponentType.Processor, ActionType.Processor {
private Set<PsiClass> myTypes;
private final String myQualifiedName;
private final PsiManager myManager;
private final GlobalSearchScope myScope;
private RegistrationTypeFinder(PsiClass psiClass, Set<PsiClass> types) {
myTypes = types;
myQualifiedName = psiClass.getQualifiedName();
myManager = psiClass.getManager();
myScope = psiClass.getResolveScope();
}
public boolean process(ComponentType type, XmlTag component, XmlTagValue impl, XmlTagValue intf) {
if (impl != null && myQualifiedName.equals(impl.getTrimmedText())) {
final PsiClass clazz = JavaPsiFacade.getInstance(myManager.getProject()).findClass(type.myClassName, myScope);
if (clazz != null) {
addType(clazz);
}
}
return true;
}
public boolean process(ActionType type, XmlTag action) {
final String actionClass = action.getAttributeValue("class");
if (actionClass != null) {
if (actionClass.trim().equals(myQualifiedName)) {
final PsiClass clazz = JavaPsiFacade.getInstance(myManager.getProject()).findClass(type.myClassName, myScope);
if (clazz != null) {
addType(clazz);
return false;
}
}
}
return true;
}
private void addType(PsiClass clazz) {
if (myTypes == null) {
//noinspection unchecked
myTypes = ContainerUtil.<PsiClass>newIdentityTroveSet(2);
}
myTypes.add(clazz);
}
public Set<PsiClass> getTypes() {
return myTypes;
}
}
}