
org.apache.commons.jexl3.internal.Debugger Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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.apache.commons.jexl3.internal;
import org.apache.commons.jexl3.JexlExpression;
import org.apache.commons.jexl3.JexlFeatures;
import org.apache.commons.jexl3.JexlInfo;
import org.apache.commons.jexl3.JexlScript;
import org.apache.commons.jexl3.parser.*;
import java.util.Collections;
import java.util.Map;
import java.util.Set;
import java.util.regex.Pattern;
/**
* Helps pinpoint the cause of problems in expressions that fail during evaluation.
*
* It rebuilds an expression string from the tree and the start/end offsets of the cause in that string.
* This implies that exceptions during evaluation do always carry the node that's causing the error.
*
* @since 2.0
*/
public class Debugger extends ParserVisitor implements JexlInfo.Detail {
/** The builder to compose messages. */
protected final StringBuilder builder = new StringBuilder();
/** The cause of the issue to debug. */
protected JexlNode cause = null;
/** The starting character location offset of the cause in the builder. */
protected int start = 0;
/** The ending character location offset of the cause in the builder. */
protected int end = 0;
/** The indentation level. */
protected int indentLevel = 0;
/** Perform indentation?. */
protected int indent = 2;
/** accept() relative depth. */
protected int depth = Integer.MAX_VALUE;
/** Arrow symbol. */
protected String arrow = "->";
/** EOL. */
protected String lf = "\n";
/** Pragmas out. */
protected boolean outputPragmas = false;
/**
* Creates a Debugger.
*/
public Debugger() {
// nothing to initialize
}
/**
* Resets this debugger state.
*/
public void reset() {
builder.setLength(0);
cause = null;
start = 0;
end = 0;
indentLevel = 0;
indent = 2;
depth = Integer.MAX_VALUE;
}
/**
* Tries (hard) to find the features used to parse a node.
* @param node the node
* @return the features or null
*/
protected JexlFeatures getFeatures(final JexlNode node) {
JexlNode walk = node;
while(walk != null) {
if (walk instanceof ASTJexlScript) {
final ASTJexlScript script = (ASTJexlScript) walk;
return script.getFeatures();
}
walk = walk.jjtGetParent();
}
return null;
}
/**
* Sets the arrow style (fat or thin) depending on features.
* @param node the node to start seeking features from.
*/
protected void setArrowSymbol(final JexlNode node) {
final JexlFeatures features = getFeatures(node);
if (features != null && features.supportsFatArrow() && !features.supportsThinArrow()) {
arrow = "=>";
} else {
arrow = "->";
}
}
/**
* Position the debugger on the root of an expression.
* @param jscript the expression
* @return true if the expression was a {@link Script} instance, false otherwise
*/
public boolean debug(final JexlExpression jscript) {
if (jscript instanceof Script) {
final Script script = (Script) jscript;
return debug(script.script);
}
return false;
}
/**
* Position the debugger on the root of a script.
* @param jscript the script
* @return true if the script was a {@link Script} instance, false otherwise
*/
public boolean debug(final JexlScript jscript) {
if (jscript instanceof Script) {
final Script script = (Script) jscript;
return debug(script.script);
}
return false;
}
/**
* Seeks the location of an error cause (a node) in an expression.
* @param node the node to debug
* @return true if the cause was located, false otherwise
*/
public boolean debug(final JexlNode node) {
return debug(node, true);
}
/**
* Seeks the location of an error cause (a node) in an expression.
* @param node the node to debug
* @param r whether we should actively find the root node of the debugged node
* @return true if the cause was located, false otherwise
*/
public boolean debug(final JexlNode node, final boolean r) {
start = 0;
end = 0;
indentLevel = 0;
setArrowSymbol(node);
if (node != null) {
builder.setLength(0);
cause = node;
// make arg cause become the root cause
JexlNode walk = node;
if (r) {
while (walk.jjtGetParent() != null) {
walk = walk.jjtGetParent();
}
}
accept(walk, null);
}
return end > 0;
}
/**
* @return The rebuilt expression
*/
@Override
public String toString() {
return builder.toString();
}
/**
* Rebuilds an expression from a JEXL node.
* @param node the node to rebuilt from
* @return the rebuilt expression
* @since 3.0
*/
public String data(final JexlNode node) {
start = 0;
end = 0;
indentLevel = 0;
setArrowSymbol(node);
if (node != null) {
builder.setLength(0);
cause = node;
accept(node, null);
}
return builder.toString();
}
/**
* @return The starting offset location of the cause in the expression
*/
@Override
public int start() {
return start;
}
/**
* @return The end offset location of the cause in the expression
*/
@Override
public int end() {
return end;
}
/**
* Lets the debugger write out pragmas if any.
* @param flag turn on or off
* @return this debugger instance
*/
public Debugger outputPragmas(final boolean flag) {
this.outputPragmas = flag;
return this;
}
/**
* Sets the indentation level.
* @param level the number of spaces for indentation, none if less or equal to zero
*/
public void setIndentation(final int level) {
indentation(level);
}
/**
* Sets the indentation level.
* @param level the number of spaces for indentation, none if less or equal to zero
* @return this debugger instance
*/
public Debugger indentation(final int level) {
indent = Math.max(level, 0);
indentLevel = 0;
return this;
}
/**
* Sets this debugger relative maximum depth.
* @param rdepth the maximum relative depth from the debugged node
* @return this debugger instance
*/
public Debugger depth(final int rdepth) {
this.depth = rdepth;
return this;
}
/**
* Sets this debugger line-feed string.
* @param lf the string used to delineate lines (usually "\" or "")
* @return this debugger instance
*/
public Debugger lineFeed(final String lf) {
this.lf = lf;
return this;
}
/**
* Checks if a child node is the cause to debug & adds its representation to the rebuilt expression.
* @param node the child node
* @param data visitor pattern argument
* @return visitor pattern value
*/
protected Object accept(final JexlNode node, final Object data) {
if (depth <= 0) {
builder.append("...");
return data;
}
if (node == cause) {
start = builder.length();
}
depth -= 1;
final Object value = node.jjtAccept(this, data);
depth += 1;
if (node == cause) {
end = builder.length();
}
return value;
}
/**
* Whether a node is a statement (vs an expression).
* @param child the node
* @return true if node is a statement
*/
private static boolean isStatement(final JexlNode child) {
return child instanceof ASTJexlScript
|| child instanceof ASTBlock
|| child instanceof ASTIfStatement
|| child instanceof ASTForeachStatement
|| child instanceof ASTWhileStatement
|| child instanceof ASTDoWhileStatement
|| child instanceof ASTAnnotation;
}
/**
* Whether a script or expression ends with a semicolumn.
* @param cs the string
* @return true if a semicolumn is the last non-whitespace character
*/
private static boolean semicolTerminated(final CharSequence cs) {
for(int i = cs.length() - 1; i >= 0; --i) {
final char c = cs.charAt(i);
if (c == ';') {
return true;
}
if (!Character.isWhitespace(c)) {
break;
}
}
return false;
}
/**
* Adds a statement node to the rebuilt expression.
* @param child the child node
* @param data visitor pattern argument
* @return visitor pattern value
*/
protected Object acceptStatement(final JexlNode child, final Object data) {
final JexlNode parent = child.jjtGetParent();
if (indent > 0 && (parent instanceof ASTBlock || parent instanceof ASTJexlScript)) {
for (int i = 0; i < indentLevel; ++i) {
for(int s = 0; s < indent; ++s) {
builder.append(' ');
}
}
}
depth -= 1;
final Object value = accept(child, data);
depth += 1;
// blocks, if, for & while don't need a ';' at end
if (!isStatement(child) && !semicolTerminated(builder)) {
builder.append(';');
if (indent > 0) {
builder.append(lf);
} else {
builder.append(' ');
}
}
return value;
}
/**
* Checks if a terminal node is the cause to debug & adds its representation to the rebuilt expression.
* @param node the child node
* @param image the child node token image (may be null)
* @param data visitor pattern argument
* @return visitor pattern value
*/
protected Object check(final JexlNode node, final String image, final Object data) {
if (node == cause) {
start = builder.length();
}
if (image != null) {
builder.append(image);
} else {
builder.append(node.toString());
}
if (node == cause) {
end = builder.length();
}
return data;
}
/**
* Checks if the children of a node using infix notation is the cause to debug, adds their representation to the
* rebuilt expression.
* @param node the child node
* @param infix the child node token
* @param paren whether the child should be parenthesized
* @param data visitor pattern argument
* @return visitor pattern value
*/
protected Object infixChildren(final JexlNode node, final String infix, final boolean paren, final Object data) {
final int num = node.jjtGetNumChildren();
if (paren) {
builder.append('(');
}
for (int i = 0; i < num; ++i) {
if (i > 0) {
builder.append(infix);
}
accept(node.jjtGetChild(i), data);
}
if (paren) {
builder.append(')');
}
return data;
}
/**
* Checks if the child of a node using prefix notation is the cause to debug, adds their representation to the
* rebuilt expression.
* @param node the node
* @param prefix the node token
* @param data visitor pattern argument
* @return visitor pattern value
*/
protected Object prefixChild(final JexlNode node, final String prefix, final Object data) {
final boolean paren = node.jjtGetChild(0).jjtGetNumChildren() > 1;
builder.append(prefix);
if (paren) {
builder.append('(');
}
accept(node.jjtGetChild(0), data);
if (paren) {
builder.append(')');
}
return data;
}
/**
* Postfix operators.
* @param node a postfix operator
* @param prefix the postfix
* @param data visitor pattern argument
* @return visitor pattern value
*/
protected Object postfixChild(final JexlNode node, final String prefix, final Object data) {
final boolean paren = node.jjtGetChild(0).jjtGetNumChildren() > 1;
if (paren) {
builder.append('(');
}
accept(node.jjtGetChild(0), data);
if (paren) {
builder.append(')');
}
builder.append(prefix);
return data;
}
@Override
protected Object visit(final ASTAddNode node, final Object data) {
return additiveNode(node, " + ", data);
}
@Override
protected Object visit(final ASTSubNode node, final Object data) {
return additiveNode(node, " - ", data);
}
/**
* Rebuilds an additive expression.
* @param node the node
* @param op the operator
* @param data visitor pattern argument
* @return visitor pattern value
*/
protected Object additiveNode(final JexlNode node, final String op, final Object data) {
// need parenthesis if not in operator precedence order
final boolean paren = node.jjtGetParent() instanceof ASTMulNode
|| node.jjtGetParent() instanceof ASTDivNode
|| node.jjtGetParent() instanceof ASTModNode;
final int num = node.jjtGetNumChildren();
if (paren) {
builder.append('(');
}
accept(node.jjtGetChild(0), data);
for (int i = 1; i < num; ++i) {
builder.append(op);
accept(node.jjtGetChild(i), data);
}
if (paren) {
builder.append(')');
}
return data;
}
@Override
protected Object visit(final ASTAndNode node, final Object data) {
return infixChildren(node, " && ", false, data);
}
@Override
protected Object visit(final ASTArrayAccess node, final Object data) {
final int num = node.jjtGetNumChildren();
for (int i = 0; i < num; ++i) {
builder.append('[');
accept(node.jjtGetChild(i), data);
builder.append(']');
}
return data;
}
@Override
protected Object visit(final ASTExtendedLiteral node, final Object data) {
builder.append("...");
return data;
}
@Override
protected Object visit(final ASTArrayLiteral node, final Object data) {
final int num = node.jjtGetNumChildren();
builder.append("[ ");
if (num > 0) {
accept(node.jjtGetChild(0), data);
for (int i = 1; i < num; ++i) {
builder.append(", ");
accept(node.jjtGetChild(i), data);
}
}
builder.append(" ]");
return data;
}
@Override
protected Object visit(final ASTRangeNode node, final Object data) {
return infixChildren(node, " .. ", false, data);
}
@Override
protected Object visit(final ASTAssignment node, final Object data) {
return infixChildren(node, " = ", false, data);
}
@Override
protected Object visit(final ASTBitwiseAndNode node, final Object data) {
return infixChildren(node, " & ", false, data);
}
@Override
protected Object visit(final ASTShiftRightNode node, final Object data) {
return infixChildren(node, " >> ", false, data);
}
@Override
protected Object visit(final ASTShiftRightUnsignedNode node, final Object data) {
return infixChildren(node, " >>> ", false, data);
}
@Override
protected Object visit(final ASTShiftLeftNode node, final Object data) {
return infixChildren(node, " << ", false, data);
}
@Override
protected Object visit(final ASTBitwiseComplNode node, final Object data) {
return prefixChild(node, "~", data);
}
@Override
protected Object visit(final ASTBitwiseOrNode node, final Object data) {
final boolean paren = node.jjtGetParent() instanceof ASTBitwiseAndNode;
return infixChildren(node, " | ", paren, data);
}
@Override
protected Object visit(final ASTBitwiseXorNode node, final Object data) {
final boolean paren = node.jjtGetParent() instanceof ASTBitwiseAndNode;
return infixChildren(node, " ^ ", paren, data);
}
@Override
protected Object visit(final ASTBlock node, final Object data) {
builder.append('{');
if (indent > 0) {
indentLevel += 1;
builder.append(lf);
} else {
builder.append(' ');
}
final int num = node.jjtGetNumChildren();
for (int i = 0; i < num; ++i) {
final JexlNode child = node.jjtGetChild(i);
acceptStatement(child, data);
}
if (indent > 0) {
indentLevel -= 1;
for (int i = 0; i < indentLevel; ++i) {
for(int s = 0; s < indent; ++s) {
builder.append(' ');
}
}
}
if (!Character.isSpaceChar(builder.charAt(builder.length() - 1))) {
builder.append(' ');
}
builder.append('}');
return data;
}
@Override
protected Object visit(final ASTDivNode node, final Object data) {
return infixChildren(node, " / ", false, data);
}
@Override
protected Object visit(final ASTEmptyFunction node, final Object data) {
builder.append("empty ");
accept(node.jjtGetChild(0), data);
return data;
}
@Override
protected Object visit(final ASTEQNode node, final Object data) {
return infixChildren(node, " == ", false, data);
}
@Override
protected Object visit(final ASTERNode node, final Object data) {
return infixChildren(node, " =~ ", false, data);
}
@Override
protected Object visit(final ASTSWNode node, final Object data) {
return infixChildren(node, " =^ ", false, data);
}
@Override
protected Object visit(final ASTEWNode node, final Object data) {
return infixChildren(node, " =$ ", false, data);
}
@Override
protected Object visit(final ASTNSWNode node, final Object data) {
return infixChildren(node, " !^ ", false, data);
}
@Override
protected Object visit(final ASTNEWNode node, final Object data) {
return infixChildren(node, " !$ ", false, data);
}
@Override
protected Object visit(final ASTFalseNode node, final Object data) {
return check(node, "false", data);
}
@Override
protected Object visit(final ASTContinue node, final Object data) {
return check(node, "continue", data);
}
@Override
protected Object visit(final ASTBreak node, final Object data) {
return check(node, "break", data);
}
@Override
protected Object visit(final ASTForeachStatement node, final Object data) {
final int form = node.getLoopForm();
builder.append("for(");
final JexlNode body;
if (form == 0) {
// for( .. : ...)
accept(node.jjtGetChild(0), data);
builder.append(" : ");
accept(node.jjtGetChild(1), data);
builder.append(") ");
body = node.jjtGetNumChildren() > 2? node.jjtGetChild(2) : null;
} else {
// for( .. ; ... ; ..)
int nc = 0;
// first child is var declaration(s)
final JexlNode vars = (form & 1) != 0 ? node.jjtGetChild(nc++) : null;
final JexlNode predicate = (form & 2) != 0 ? node.jjtGetChild(nc++) : null;
// the loop step
final JexlNode step = (form & 4) != 0 ? node.jjtGetChild(nc++) : null;
// last child is body
body = (form & 8) != 0 ? node.jjtGetChild(nc) : null;
if (vars != null) {
accept(vars, data);
}
builder.append("; ");
if (predicate != null) {
accept(predicate, data);
}
builder.append("; ");
if (step != null) {
accept(step, data);
}
builder.append(") ");
}
// the body
if (body != null) {
accept(body, data);
} else {
builder.append(';');
}
return data;
}
@Override
protected Object visit(final ASTGENode node, final Object data) {
return infixChildren(node, " >= ", false, data);
}
@Override
protected Object visit(final ASTGTNode node, final Object data) {
return infixChildren(node, " > ", false, data);
}
/** Checks identifiers that contain spaces or punctuation
* (but underscore, at-sign, sharp-sign and dollar).
*/
protected static final Pattern QUOTED_IDENTIFIER =
Pattern.compile("[\\s]|[\\p{Punct}&&[^@#$_]]");
/**
* Checks whether an identifier should be quoted or not.
* @param str the identifier
* @return true if needing quotes, false otherwise
*/
protected boolean needQuotes(final String str) {
return QUOTED_IDENTIFIER.matcher(str).find()
|| "size".equals(str)
|| "empty".equals(str);
}
@Override
protected Object visit(final ASTIdentifier node, final Object data) {
final String ns = node.getNamespace();
final String image = StringParser.escapeIdentifier(node.getName());
if (ns == null) {
return check(node, image, data);
}
final String nsid = StringParser.escapeIdentifier(ns) + ":" + image;
return check(node, nsid, data);
}
@Override
protected Object visit(final ASTIdentifierAccess node, final Object data) {
builder.append(node.isSafe() ? "?." : ".");
final String image = node.getName();
if (node.isExpression()) {
builder.append('`');
builder.append(image.replace("`", "\\`"));
builder.append('`');
} else if (needQuotes(image)) {
// quote it
builder.append('\'');
builder.append(image.replace("'", "\\'"));
builder.append('\'');
} else {
builder.append(image);
}
return data;
}
@Override
protected Object visit(final ASTIfStatement node, final Object data) {
final int numChildren = node.jjtGetNumChildren();
// if (...) ...
builder.append("if (");
accept(node.jjtGetChild(0), data);
builder.append(") ");
acceptStatement(node.jjtGetChild(1), data);
//.. else if (...) ...
for(int c = 2; c < numChildren - 1; c += 2) {
builder.append(" else if (");
accept(node.jjtGetChild(c), data);
builder.append(") ");
acceptStatement(node.jjtGetChild(c + 1), data);
}
// else... (if odd)
if ((numChildren & 1) == 1) {
builder.append(" else ");
acceptStatement(node.jjtGetChild(numChildren - 1), data);
}
return data;
}
@Override
protected Object visit(final ASTNumberLiteral node, final Object data) {
return check(node, node.toString(), data);
}
/**
* A pseudo visitor for parameters.
* @param p the parameter name
* @param data the visitor argument
* @return the parameter name to use
*/
protected String visitParameter(final String p, final Object data) {
return p;
}
private static boolean isLambdaExpr(final ASTJexlLambda lambda) {
return lambda.jjtGetNumChildren() == 1 && !isStatement(lambda.jjtGetChild(0));
}
/**
* Stringifies the pragmas.
* @param builder where to stringify
* @param pragmas the pragmas, may be null
*/
private static void writePragmas(final StringBuilder builder, final Map pragmas) {
if (pragmas != null) {
for (final Map.Entry pragma : pragmas.entrySet()) {
final String key = pragma.getKey();
final Object value = pragma.getValue();
final Set
© 2015 - 2025 Weber Informatics LLC | Privacy Policy