/*
* Sonar PHP Plugin
* Copyright (C) 2010 Sonar PHP Plugin
* dev@sonar.codehaus.org
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 3 of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02
*/
package org.sonar.plugins.php.codesniffer;
import static org.apache.commons.lang.StringUtils.isBlank;
import static org.sonar.plugins.php.codesniffer.PhpCodeSnifferRuleRepository.PHPCS_REPOSITORY_KEY;
import java.io.IOException;
import java.io.Reader;
import org.jdom.Document;
import org.jdom.Element;
import org.jdom.JDOMException;
import org.jdom.Namespace;
import org.jdom.input.SAXBuilder;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.sonar.api.profiles.RulesProfile;
import org.sonar.api.rules.ActiveRule;
import org.sonar.api.rules.Rule;
import org.sonar.api.rules.RuleFinder;
import org.sonar.api.rules.RuleQuery;
import org.sonar.api.utils.ValidationMessages;
import org.sonar.plugins.php.api.Php;
import org.sonar.plugins.php.core.PhpProfileImporter;
import org.sonar.plugins.php.pmd.xml.PmdProperty;
import org.sonar.plugins.php.pmd.xml.PmdRule;
import org.sonar.plugins.php.pmd.xml.PmdRuleset;
/**
* @author Akram Ben Aissi
*
*/
public class PhpCodeSnifferProfileImporter extends PhpProfileImporter {
private static final Logger LOG = LoggerFactory.getLogger(PhpCodeSnifferProfileImporter.class);
/**
*
*/
private final RuleFinder ruleFinder;
/**
*
*/
private final PhpCodeSnifferPriorityMapper mapper;
/**
* @param ruleFinder
*/
public PhpCodeSnifferProfileImporter(RuleFinder ruleFinder, PhpCodeSnifferPriorityMapper mapper) {
super(PhpCodeSnifferRuleRepository.PHPCS_REPOSITORY_KEY, PhpCodeSnifferRuleRepository.PHPCS_REPOSITORY_NAME);
setSupportedLanguages(Php.KEY);
this.ruleFinder = ruleFinder;
this.mapper = mapper;
}
/**
* @see org.sonar.api.profiles.ProfileImporter#importProfile(java.io.Reader, org.sonar.api.utils.ValidationMessages)
*/
@Override
public RulesProfile importProfile(Reader pmdConfigurationFile, ValidationMessages messages) {
PmdRuleset ruleSet = parseRuleset(pmdConfigurationFile, messages);
return createRuleProfile(ruleSet, messages);
}
/**
* @param pmdRuleset
* @param messages
* @return
*/
protected RulesProfile createRuleProfile(PmdRuleset pmdRuleset, ValidationMessages messages) {
RulesProfile profile = RulesProfile.create(pmdRuleset.getName(), Php.KEY);
for (PmdRule pmdRule : pmdRuleset.getPmdRules()) {
boolean isRuleValid = true;
String key = pmdRule.getRef();
if (key == null) {
messages.addWarningText("A rule without 'ref' attribute can't be imported. see '" + pmdRule.getClazz() + "'");
isRuleValid = false;
}
// Attention: rule is retrieved using the key field which different of what is done on PmdImporter that uses configKey.
Rule rule = ruleFinder.find(RuleQuery.create().withRepositoryKey(PHPCS_REPOSITORY_KEY).withKey(key));
if (rule == null) {
StringBuilder message = new StringBuilder("Unable to import unknown PMD rule '");
message.append(key).append("' consider adding an extension in sonar extenions directory");
messages.addWarningText(message.toString());
isRuleValid = false;
}
ActiveRule activeRule = profile.activateRule(rule, mapper.from(pmdRule.getPriority()));
if (isRuleValid && pmdRule.getProperties() != null) {
createRule(messages, pmdRule, key, rule, activeRule);
}
}
return profile;
}
/**
* @param messages
* @param pmdRule
* @param key
* @param rule
* @param activeRule
*/
private void createRule(ValidationMessages messages, PmdRule pmdRule, String key, Rule rule, ActiveRule activeRule) {
for (PmdProperty prop : pmdRule.getProperties()) {
String name = prop.getName();
if (rule.getParam(name) != null) {
String value = prop.getValue();
String ruleValue = prop.isCdataValue() && isBlank(value) ? prop.getCdataValue() : value;
activeRule.setParameter(name, ruleValue);
} else {
StringBuilder message = new StringBuilder("The property '").append(name);
message.append("' is not supported in the PhpCodeSniffer rule: ").append(key);
messages.addWarningText(message.toString());
}
}
}
/**
* @param pmdConfigurationFile
* @param messages
* @return
*/
protected PmdRuleset parseRuleset(Reader pmdConfigurationFile, ValidationMessages messages) {
try {
SAXBuilder parser = new SAXBuilder();
Document dom = parser.build(pmdConfigurationFile);
Element ruleSetNode = dom.getRootElement();
Namespace namespace = ruleSetNode.getNamespace();
PmdRuleset pmdResultset = new PmdRuleset(ruleSetNode.getAttributeValue("name"));
for (Element ruleNode : getChildren(ruleSetNode, "rule", namespace)) {
PmdRule pmdRule = new PmdRule(ruleNode.getAttributeValue("ref"));
pmdRule.setClazz(ruleNode.getAttributeValue("class"));
pmdRule.setName(ruleNode.getAttributeValue("name"));
pmdRule.setMessage(ruleNode.getAttributeValue("message"));
parsePmdPriority(ruleNode, pmdRule, namespace);
parsePmdProperties(ruleNode, pmdRule, namespace);
pmdResultset.addRule(pmdRule);
}
return pmdResultset;
} catch (JDOMException e) {
return emptyRuleSetAndLogMessage(messages, e);
} catch (IOException e) {
return emptyRuleSetAndLogMessage(messages, e);
}
}
/**
* @param messages
* @param e
*/
private PmdRuleset emptyRuleSetAndLogMessage(ValidationMessages messages, Exception e) {
String errorMessage = "The PMD configuration file is not valid";
messages.addErrorText(errorMessage + " : " + e.getMessage());
LOG.error(errorMessage, e);
return new PmdRuleset();
}
}