
com.imsweb.validation.internal.EditCodeVisitorSupport Maven / Gradle / Ivy
/*
* Copyright (C) 2008 Information Management Services, Inc.
*/
package com.imsweb.validation.internal;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.commons.lang3.StringUtils;
import org.codehaus.groovy.ast.CodeVisitorSupport;
import org.codehaus.groovy.ast.expr.ArgumentListExpression;
import org.codehaus.groovy.ast.expr.BinaryExpression;
import org.codehaus.groovy.ast.expr.ClosureExpression;
import org.codehaus.groovy.ast.expr.ConstantExpression;
import org.codehaus.groovy.ast.expr.DeclarationExpression;
import org.codehaus.groovy.ast.expr.Expression;
import org.codehaus.groovy.ast.expr.MethodCallExpression;
import org.codehaus.groovy.ast.expr.PropertyExpression;
import org.codehaus.groovy.ast.expr.VariableExpression;
import org.codehaus.groovy.ast.stmt.ForStatement;
import com.imsweb.validation.ValidationEngine;
import com.imsweb.validation.ValidatorContextFunctions;
import com.imsweb.validation.ValidatorServices;
import com.imsweb.validation.shared.ContextFunctionAliasAnnotation;
/**
* Parses groovy edits to gather used properties...
*
* Created on Jan 15, 2008 by depryf
* @author depryf
*/
public class EditCodeVisitorSupport extends CodeVisitorSupport {
// generic methods that can be used at the end of the path as a field (something like 'object.property.trim')
private static final List _METHODS_AS_FIELDS = Arrays.asList("size", "empty", "trim", "toUpperCase", "toLowerCase");
/**
* This will contain the returned properties that have been identified during the parsing
*/
protected Set _properties;
/**
* These are potential context keys that have been identified during the parsing - potential context keys are the validator
* context entries that are required by the rule being parsed. They are potential because the parsing is not perfect and we
* might have some entries that are actually not a context key. In that case the code using this entries will just ignore it.
*
* In a future version, the "Context." prefix will be required for any context, and the parsing will actually be accurate.
*
* The parsed context never contain the "Functions" nor "Context" prefixes.
*/
protected Set _contextEntries;
/**
* This will contain any lookup used in the expression (those are used with the call "fetchLookup")
*/
protected Set _lookups;
/**
* These are variable aliases identified during the parsing (used internally only)
*/
protected Map _variableAliases;
/**
* Keep track of the defined variables so we don't mistake them for context entries
*/
protected List _defVariables;
/**
* Do we need to ensure the 'def' keyword is properly used?
*/
protected boolean _forceDefKeyword;
/**
* Constructor
*
* Created on Jan 15, 2008 by depryf
* @param properties place holder for gathered properties (can be null)
* @param contextEntries place holder for gathered context entries (can be null)
* @param lookups place holder for gathered lookups (can be null)
* @param forceDefKeyword if true and a variable is defined without the def keyword, then an exception will be raised
*/
public EditCodeVisitorSupport(Set properties, Set contextEntries, Set lookups, boolean forceDefKeyword) {
_properties = properties == null ? new HashSet<>() : properties;
_contextEntries = contextEntries == null ? new HashSet<>() : contextEntries;
_lookups = lookups == null ? new HashSet<>() : lookups;
_forceDefKeyword = forceDefKeyword;
_variableAliases = new HashMap<>();
_defVariables = new ArrayList<>();
}
@Override
public void visitPropertyExpression(PropertyExpression expression) {
// if (ctc.primarySite == null) {...}
// if (treatmentProcedure.person.nameLast == null) {...}
if (expression.getProperty() instanceof ConstantExpression) {
String prop = expression.getProperty().getText();
String[] parts = StringUtils.split(StringUtils.replace(expression.getObjectExpression().getText(), "?", ""), '.');
if (parts.length > 0) {
if (ValidationEngine.VALIDATOR_CONTEXT_KEY.equals(parts[0]))
_contextEntries.add(prop);
else {
// have to use special cases for lines/untrimmedlines because the root of the path for those is a list instead of an object...
StringBuilder buf = new StringBuilder();
if (parts.length > 1 && parts[0].equals("lines") && parts[1].startsWith("get("))
buf.append("line");
else if (parts.length > 1 && parts[0].equals("untrimmedlines") && parts[1].startsWith("get("))
buf.append("untrimmedline");
else {
if (_variableAliases.containsKey(parts[0]))
buf.append(_variableAliases.get(parts[0]));
else if (parts[0].startsWith("lines["))
buf.append("line");
else if (parts[0].startsWith("untrimmedlines["))
buf.append("untrimmedline");
else
buf.append(parts[0]);
for (int i = 1; i < parts.length; i++)
buf.append(".").append(parts[i]);
}
buf.append(".").append(prop);
String[] parts2 = StringUtils.split(buf.toString(), '.');
if (ValidatorServices.getInstance().getJavaPathForAlias(parts2[0]) != null) {
StringBuilder innerBuf = new StringBuilder(parts2[0]);
for (int i = 1; i < parts2.length; i++)
if (i != parts2.length - 1 || !_METHODS_AS_FIELDS.contains(parts2[i]))
innerBuf.append(".").append(parts2[i]);
String property = innerBuf.toString();
if (ValidatorServices.getInstance().getAliasForJavaPath(property) == null) {
_properties.add(innerBuf.toString());
return;
}
}
}
}
}
super.visitPropertyExpression(expression);
}
@Override
public void visitVariableExpression(VariableExpression expression) {
// this is the old way of accessing context entries (without a prefix); support for this will be removed eventually
String name = expression.getName();
if (!"this".equals(name) && !_defVariables.contains(name) && ValidatorServices.getInstance().getJavaPathForAlias(name) == null)
if (!isInternalContextName(name))
_contextEntries.add(name);
}
@Override
public void visitDeclarationExpression(DeclarationExpression expression) {
// def site1 = line1.primarySite
// def reg = facilityAdmission.registryData
// def ctc1 = patientc.ctcs.get(index1)
// this is not perfect because it doesn't take into account the limited scope, for example a def inside a loop...
_defVariables.add(expression.getLeftExpression().getText());
if (expression.getRightExpression() instanceof MethodCallExpression) {
MethodCallExpression call = (MethodCallExpression)expression.getRightExpression();
for (Method m : ValidatorContextFunctions.getInstance().getClass().getMethods()) {
if (m.getName().equals(call.getMethodAsString()) && m.getAnnotation(ContextFunctionAliasAnnotation.class) != null) {
_variableAliases.put(expression.getLeftExpression().getText(), m.getAnnotation(ContextFunctionAliasAnnotation.class).value());
break;
}
}
expression.getRightExpression().visit(this);
}
else if (expression.getRightExpression() instanceof VariableExpression) {
_variableAliases.put(expression.getLeftExpression().getText(), expression.getRightExpression().getText());
}
else if (expression.getRightExpression() instanceof PropertyExpression) {
_variableAliases.put(expression.getLeftExpression().getText(), expression.getRightExpression().getText());
expression.getRightExpression().visit(this);
}
else
expression.getRightExpression().visit(this);
}
@Override
public void visitForLoop(ForStatement forLoop) {
// for (ctc : patient.ctcs) {...}
String alias = getAliasForPartialPath(forLoop.getCollectionExpression().getText());
if (alias != null) {
_variableAliases.put(forLoop.getVariable().getName(), alias);
forLoop.getLoopBlock().visit(this);
_variableAliases.remove(forLoop.getVariable().getName());
}
else
forLoop.getLoopBlock().visit(this);
}
@Override
public void visitMethodCallExpression(MethodCallExpression call) {
// if (Functions.between(ctc.primarySite, 'C123', 'C456') {...}
// patient.ctcs.each {...}
// patient.ctcs.each {alias -> ...}
// def ctc1 = patientc.ctcs.get(index1)
String method = call.getMethodAsString();
String alias = getAliasForPartialPath(call.getObjectExpression().getText());
if (alias != null) {
if (call.getArguments() instanceof ArgumentListExpression) {
ArgumentListExpression list = (ArgumentListExpression)call.getArguments();
if (list.getExpressions().size() == 1 && list.getExpression(0) instanceof ClosureExpression) {
ClosureExpression closure = (ClosureExpression)list.getExpression(0);
String groovyAlias = closure.getParameters().length > 0 ? closure.getParameters()[0].getName() : "it";
_variableAliases.put(groovyAlias, alias);
call.getArguments().visit(this);
_variableAliases.remove(groovyAlias);
return;
}
else if ("get".equals(method)) {
if (!_defVariables.isEmpty())
_variableAliases.put(_defVariables.get(_defVariables.size() - 1), alias);
}
}
}
String caller = call.getObjectExpression().getText();
// methods can be called on context (if the context is a closure for example)
if (ValidationEngine.VALIDATOR_CONTEXT_KEY.equals(caller))
_contextEntries.add(method);
// any method called on "this" is a context entry (this is the old way of calling contexts, without a prefix...)
if ((call.getObjectExpression() instanceof VariableExpression && "this".equals(caller)) && !_defVariables.contains(method))
if (ValidatorServices.getInstance().getJavaPathForAlias(method) == null && !isInternalContextName(method))
_contextEntries.add(method);
// the calling object could also be a context entry: ARRAY.contains(...)
if ((call.getObjectExpression() instanceof VariableExpression && !"this".equals(caller)) && !_defVariables.contains(caller))
if (ValidatorServices.getInstance().getJavaPathForAlias(caller) == null && !isInternalContextName(caller))
_contextEntries.add(caller);
if ("fetchLookup".equals(method)) {
// format is "(lkup_id)", so we just need to remove the parenthesis
String rawLkupId = call.getArguments().getText();
_lookups.add(rawLkupId.substring(1, rawLkupId.length() - 1));
}
super.visitMethodCallExpression(call);
}
@Override
public void visitBinaryExpression(BinaryExpression expression) {
if (_forceDefKeyword) {
Expression left = expression.getLeftExpression();
if (!_defVariables.contains(left.getText()) && left instanceof VariableExpression && expression.getOperation().getText().equals("="))
throw new RuntimeException("Variable '" + expression.getLeftExpression().getText() + "' is not defined using the 'def' keyword");
}
// def line1 = lines[index]
// def ctc1 = patient.ctcs[index]
if (expression.getOperation() != null && expression.getOperation().getText().equals("[")) {
String alias = getAliasForPartialPath(expression.getLeftExpression().getText());
if (alias != null)
if (!_defVariables.isEmpty())
_variableAliases.put(_defVariables.get(_defVariables.size() - 1), alias);
}
super.visitBinaryExpression(expression);
}
// helper
protected String getAliasForPartialPath(String path) {
String alias = null;
String[] parts = StringUtils.split(path, '.');
if (parts.length > 0) {
String javaPath = ValidatorServices.getInstance().getJavaPathForAlias(parts[0]);
if (javaPath != null) {
StringBuilder buf = new StringBuilder(javaPath);
for (int i = 1; i < parts.length; i++)
buf.append(".").append(parts[i]);
alias = ValidatorServices.getInstance().getAliasForJavaPath(buf.toString());
// I know, this is a hack, but I can't make it work otherwise; this is due to the fact that normally, the
// top level of an entity (so validatable root) is an entity (for example a patient), but for the SEER edits,
// it is a collection (lines, which is a collection of line). That messes up a lot of things...
if ("lines".equals(alias))
alias = "line";
else if ("untrimmedlines".equals(alias))
alias = "untrimmedline";
}
}
return alias;
}
// helper
@SuppressWarnings("BooleanMethodIsAlwaysInverted")
protected boolean isInternalContextName(String name) {
return ValidationEngine.VALIDATOR_FUNCTIONS_KEY.equals(name) || ValidationEngine.VALIDATOR_CONTEXT_KEY.equals(name);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy