blob: b006ac1c2f174f255964272c60a7f67a195b2083 [file] [log] [blame]
package com.jetbrains.python.psi.impl;
import com.jetbrains.python.psi.*;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
/**
* Evaluator tht supports expression of any type (it evaluates any expression, concatenating several expressions to list etc)
*/
public class PyAnyExpressionEvaluator extends PyEvaluator {
private final boolean myEvalSequence;
/**
* @param evalSequence evaluate sequencies ((a,b) + (c,d) == (a,b,c,d)) or store them as single ((a,b) + (c,d) = [(a,b), (c,d)])
*/
public PyAnyExpressionEvaluator(final boolean evalSequence) {
myEvalSequence = evalSequence;
}
@NotNull
@Override
public Object evaluate(final PyExpression expr) {
final Object evaluate = super.evaluate(expr);
return ((evaluate != null) ? evaluate : expr);
}
@Override
public Object concatenate(final Object lhs, final Object rhs) {
final Object evaluate = super.concatenate(lhs, rhs);
return ((evaluate != null) ? evaluate : Arrays.asList(lhs, rhs));
}
@Override
protected Object evaluateReferenceExpression(final PyReferenceExpression expr) {
final Object evaluate = super.evaluateReferenceExpression(expr);
return ((evaluate != null) ? evaluate : expr);
}
@Override
protected Object evaluateCall(final PyCallExpression call) {
final Object evaluate = super.evaluateCall(call);
return ((evaluate != null) ? evaluate : call);
}
@Override
protected Object evaluateSequenceExpression(final PySequenceExpression expr) {
return myEvalSequence ? super.evaluateSequenceExpression(expr) : expr;
}
/**
* Evaluates expression to single element
* @param expression exp to eval
* @param aClass expected class
* @param <T> expected class
* @return instance of aClass, or null if failed to eval
*/
@Nullable
public static <T> T evaluateOne(@NotNull final PyExpression expression, @NotNull final Class<T> aClass) {
final PyAnyExpressionEvaluator evaluator = new PyAnyExpressionEvaluator(false);
final Object evaluate = evaluator.evaluate(expression);
final T resultSingle = PyUtil.as(evaluate, aClass);
if (resultSingle != null) {
return resultSingle;
}
final List<?> resultMultiple = PyUtil.as(evaluate, List.class);
if ((resultMultiple != null) && !resultMultiple.isEmpty()) {
return PyUtil.as(resultMultiple.get(0), aClass);
}
return null;
}
/**
* Evaluates expression to string
* @param expression exp to eval
* @return string, or null if failed to eval
*/
@Nullable
public static String evaluateString(@NotNull final PyExpression expression) {
return PyUtil.as(new PyAnyExpressionEvaluator(false).evaluate(expression), String.class);
}
/**
* Evaluates expression as list of values
* @param expression exp to eval
* @param aClass expected element class
* @param <T> expected element class
* @return a list of elements of expected type
*/
@NotNull
public static <T>List<T> evaluateIterable(@NotNull final PyExpression expression, @NotNull final Class<T> aClass) {
final PyAnyExpressionEvaluator evaluator = new PyAnyExpressionEvaluator(true);
final Object evaluate = evaluator.evaluate(expression);
final T resultSingle = PyUtil.as(evaluate, aClass);
if (resultSingle != null) {
return Collections.singletonList(resultSingle);
}
return PyUtil.asList(PyUtil.as(evaluate, List.class), aClass);
}
}