All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.puppycrawl.tools.checkstyle.checks.coding.MatchXpathCheck Maven / Gradle / Ivy

Go to download

Checkstyle is a development tool to help programmers write Java code that adheres to a coding standard

There is a newer version: 10.20.1
Show newest version
///////////////////////////////////////////////////////////////////////////////////////////////
// checkstyle: Checks Java source code and other text files for adherence to a set of rules.
// Copyright (C) 2001-2024 the original author or authors.
//
// This library 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 2.1 of the License, or (at your option) any later version.
//
// This library 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 library; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
///////////////////////////////////////////////////////////////////////////////////////////////

package com.puppycrawl.tools.checkstyle.checks.coding;

import java.util.List;
import java.util.stream.Collectors;

import com.puppycrawl.tools.checkstyle.StatelessCheck;
import com.puppycrawl.tools.checkstyle.api.AbstractCheck;
import com.puppycrawl.tools.checkstyle.api.DetailAST;
import com.puppycrawl.tools.checkstyle.utils.CommonUtil;
import com.puppycrawl.tools.checkstyle.xpath.AbstractNode;
import com.puppycrawl.tools.checkstyle.xpath.RootNode;
import net.sf.saxon.Configuration;
import net.sf.saxon.om.Item;
import net.sf.saxon.sxpath.XPathDynamicContext;
import net.sf.saxon.sxpath.XPathEvaluator;
import net.sf.saxon.sxpath.XPathExpression;
import net.sf.saxon.trans.XPathException;

/**
 * 

* Evaluates Xpath query and report violation on all matching AST nodes. This check allows * user to implement custom checks using Xpath. If Xpath query is not specified explicitly, * then the check does nothing. *

*

* It is recommended to define custom message for violation to explain what is not allowed and what * to use instead, default message might be too abstract. To customize a message you need to * add {@code message} element with matchxpath.match as {@code key} attribute and * desired message as {@code value} attribute. *

*

* Please read more about Xpath syntax at * Xpath Syntax. * Information regarding Xpath functions can be found at * * XSLT/XPath Reference. * Note, that @text attribute can be used only with token types that are listed in * * XpathUtil. *

*
    *
  • * Property {@code query} - Specify Xpath query. * Type is {@code java.lang.String}. * Default value is {@code ""}. *
  • *
*

* Parent is {@code com.puppycrawl.tools.checkstyle.TreeWalker} *

*

* Violation Message Keys: *

*
    *
  • * {@code matchxpath.match} *
  • *
* * @since 8.39 */ @StatelessCheck public class MatchXpathCheck extends AbstractCheck { /** * A key is pointing to the warning message text provided by user. */ public static final String MSG_KEY = "matchxpath.match"; /** Specify Xpath query. */ private String query = ""; /** Xpath expression. */ private XPathExpression xpathExpression; /** * Setter to specify Xpath query. * * @param query Xpath query. * @throws IllegalStateException if creation of xpath expression fails * @since 8.39 */ public void setQuery(String query) { this.query = query; if (!query.isEmpty()) { try { final XPathEvaluator xpathEvaluator = new XPathEvaluator(Configuration.newConfiguration()); xpathExpression = xpathEvaluator.createExpression(query); } catch (XPathException ex) { throw new IllegalStateException("Creating Xpath expression failed: " + query, ex); } } } @Override public int[] getDefaultTokens() { return getRequiredTokens(); } @Override public int[] getAcceptableTokens() { return getRequiredTokens(); } @Override public int[] getRequiredTokens() { return CommonUtil.EMPTY_INT_ARRAY; } @Override public boolean isCommentNodesRequired() { return true; } @Override public void beginTree(DetailAST rootAST) { if (!query.isEmpty()) { final List matchingNodes = findMatchingNodesByXpathQuery(rootAST); matchingNodes.forEach(node -> log(node, MSG_KEY)); } } /** * Find nodes that match query. * * @param rootAST root node * @return list of matching nodes * @throws IllegalStateException if evaluation of xpath query fails */ private List findMatchingNodesByXpathQuery(DetailAST rootAST) { try { final RootNode rootNode = new RootNode(rootAST); final XPathDynamicContext xpathDynamicContext = xpathExpression.createDynamicContext(rootNode); final List matchingItems = xpathExpression.evaluate(xpathDynamicContext); return matchingItems.stream() .map(item -> (DetailAST) ((AbstractNode) item).getUnderlyingNode()) .collect(Collectors.toUnmodifiableList()); } catch (XPathException ex) { throw new IllegalStateException("Evaluation of Xpath query failed: " + query, ex); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy