replace field string list with SelectableExpression list in SelectExpression
This commit is contained in:
@@ -3,6 +3,7 @@ package jef;
|
||||
import jef.expressions.Expression;
|
||||
import jef.expressions.SelectExpression;
|
||||
import jef.expressions.TableExpression;
|
||||
import jef.expressions.selectable.DatabaseSelectAllExpression;
|
||||
|
||||
import java.io.Serializable;
|
||||
import java.util.Iterator;
|
||||
@@ -23,7 +24,7 @@ public class DBSet<T extends Serializable> implements Queryable<T> {
|
||||
|
||||
@Override
|
||||
public Expression getExpression() {
|
||||
return new SelectExpression(List.of("*"), new TableExpression(table), "");
|
||||
return new SelectExpression(List.of(new DatabaseSelectAllExpression()), new TableExpression(table), "");
|
||||
}
|
||||
|
||||
@Override
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
package jef;
|
||||
|
||||
import jef.expressions.Expression;
|
||||
import jef.operations.CountOp;
|
||||
import jef.operations.FilterOp;
|
||||
import jef.serializable.SerializablePredicate;
|
||||
|
||||
@@ -52,7 +53,7 @@ public interface Queryable<T extends Serializable> {
|
||||
|
||||
//stream
|
||||
default Queryable<T> filter(SerializablePredicate<? super T> predicate) {
|
||||
return new FilterOp<T>(this, predicate);
|
||||
return new FilterOp<>(this, predicate);
|
||||
}
|
||||
|
||||
// default <R extends Serializable> Queryable<R> map(Function<? super T, ? extends R> function) {
|
||||
|
||||
@@ -3,7 +3,7 @@ package jef.asm;
|
||||
import jef.expressions.BinaryExpression;
|
||||
import jef.expressions.ConstantExpression;
|
||||
import jef.expressions.Expression;
|
||||
import jef.expressions.FieldExpression;
|
||||
import jef.expressions.IntermediateFieldExpression;
|
||||
import jef.expressions.NullExpression;
|
||||
import jef.expressions.ParameterExpression;
|
||||
import jef.expressions.SelectExpression;
|
||||
@@ -11,6 +11,7 @@ import jef.expressions.TableExpression;
|
||||
import jef.expressions.TernaryExpression;
|
||||
import jef.expressions.UnaryExpression;
|
||||
import jef.expressions.WhereExpression;
|
||||
import jef.expressions.selectable.DatabaseSelectAllExpression;
|
||||
import lombok.ToString;
|
||||
import org.objectweb.asm.Attribute;
|
||||
import org.objectweb.asm.Label;
|
||||
@@ -66,7 +67,7 @@ class FilterMethodVisitor extends MethodVisitor {
|
||||
var v = varStack.pop();
|
||||
if (v instanceof ParameterExpression p) {
|
||||
if (p.isInput()) {
|
||||
varStack.push(new FieldExpression(name, descriptor));
|
||||
varStack.push(new IntermediateFieldExpression(name, descriptor));
|
||||
} else {
|
||||
throw new RuntimeException("field insn: unsupported GETFIELD expression");
|
||||
}
|
||||
@@ -106,7 +107,7 @@ class FilterMethodVisitor extends MethodVisitor {
|
||||
var v = varStack.pop();
|
||||
if (v instanceof ParameterExpression p) {
|
||||
if (p.isInput()) {
|
||||
varStack.push(new FieldExpression(field.get().getName(), descriptor.substring(2)));
|
||||
varStack.push(new IntermediateFieldExpression(field.get().getName(), descriptor.substring(2)));
|
||||
} else {
|
||||
throw new RuntimeException("method insn: getter support only to predicate parameter");
|
||||
}
|
||||
@@ -463,7 +464,7 @@ class FilterMethodVisitor extends MethodVisitor {
|
||||
|
||||
private void debugExpr() {
|
||||
if (!varStack.isEmpty()) {
|
||||
System.out.println("-------------------> " + new WhereExpression(new SelectExpression(List.of("*"), new TableExpression("dummy"), ""), varStack.peek()));
|
||||
System.out.println("-------------------> " + new WhereExpression(new SelectExpression(List.of(new DatabaseSelectAllExpression()), new TableExpression("dummy"), ""), varStack.peek()));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -13,6 +13,7 @@ public interface Expression {
|
||||
BINARY,
|
||||
CONSTANT,
|
||||
FIELD,
|
||||
INTERMEDIATE_FIELD,
|
||||
NULL,
|
||||
OR,
|
||||
PARAMETER,
|
||||
|
||||
@@ -1,14 +1,19 @@
|
||||
package jef.expressions;
|
||||
|
||||
import jef.expressions.selectable.SelectableExpression;
|
||||
import lombok.Getter;
|
||||
|
||||
@Getter
|
||||
public class FieldExpression extends ConstantExpression implements Expression {
|
||||
public class FieldExpression extends ConstantExpression implements SelectableExpression, Expression {
|
||||
private final String schema;
|
||||
private final String table;
|
||||
private final String name;
|
||||
private final String classDescriptor;
|
||||
|
||||
public FieldExpression(String name, String classDescriptor) {
|
||||
public FieldExpression(String schema, String table, String name, String classDescriptor) {
|
||||
super(name);
|
||||
this.schema = schema;
|
||||
this.table = table;
|
||||
this.name = name;
|
||||
this.classDescriptor = classDescriptor;
|
||||
}
|
||||
@@ -20,6 +25,8 @@ public class FieldExpression extends ConstantExpression implements Expression {
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return name;
|
||||
return (schema != null && !schema.isEmpty() ? "`" + schema + "`." : "")
|
||||
+ (table != null && !table.isEmpty() ? "`" + table + "`." : "")
|
||||
+ "`" + name + "`";
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,26 @@
|
||||
package jef.expressions;
|
||||
|
||||
import jef.expressions.selectable.SelectableExpression;
|
||||
import lombok.Getter;
|
||||
|
||||
@Getter
|
||||
public class IntermediateFieldExpression extends ConstantExpression implements SelectableExpression, Expression {
|
||||
private final String name;
|
||||
private final String classDescriptor;
|
||||
|
||||
public IntermediateFieldExpression(String name, String classDescriptor) {
|
||||
super(name);
|
||||
this.name = name;
|
||||
this.classDescriptor = classDescriptor;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Type getType() {
|
||||
return Type.INTERMEDIATE_FIELD;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return name;
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,6 @@
|
||||
package jef.expressions;
|
||||
|
||||
import jef.expressions.selectable.SelectableExpression;
|
||||
import lombok.AllArgsConstructor;
|
||||
import lombok.Getter;
|
||||
|
||||
@@ -9,7 +10,7 @@ import java.util.stream.Collectors;
|
||||
@Getter
|
||||
@AllArgsConstructor
|
||||
public class SelectExpression implements Expression {
|
||||
private final List<String> fields;
|
||||
private final List<SelectableExpression> fields;
|
||||
private final Expression from;
|
||||
private final String fromAlias;
|
||||
|
||||
@@ -25,8 +26,9 @@ public class SelectExpression implements Expression {
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "SELECT " + fields.stream().map(e -> e.equals("*") ? e : "`" + e + "`").collect(Collectors.joining(", ")) + " FROM "
|
||||
return "SELECT " + fields.stream().map(SelectableExpression::toString).collect(Collectors.joining(", ")) + " FROM "
|
||||
+ (!(from instanceof TableExpression) ? "(" + from + ")" : from)
|
||||
+ ((fromAlias == null || fromAlias.isBlank()) ? "" : " " + fromAlias);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
@@ -5,6 +5,7 @@ import jef.expressions.BinaryExpression;
|
||||
import jef.expressions.ConstantExpression;
|
||||
import jef.expressions.Expression;
|
||||
import jef.expressions.FieldExpression;
|
||||
import jef.expressions.IntermediateFieldExpression;
|
||||
import jef.expressions.NullExpression;
|
||||
import jef.expressions.OrExpression;
|
||||
import jef.expressions.ParameterExpression;
|
||||
@@ -25,6 +26,7 @@ public abstract class ExpressionModifier {
|
||||
case BINARY -> modifyBinary((BinaryExpression) expr);
|
||||
case CONSTANT -> modifyConstant((ConstantExpression) expr);
|
||||
case FIELD -> modifyField((FieldExpression) expr);
|
||||
case INTERMEDIATE_FIELD -> modifyIntermediateField((IntermediateFieldExpression) expr);
|
||||
case NULL -> modifyNull((NullExpression) expr);
|
||||
case OR -> modifyOr((OrExpression) expr);
|
||||
case PARAMETER -> modifyParameter((ParameterExpression) expr);
|
||||
@@ -57,6 +59,10 @@ public abstract class ExpressionModifier {
|
||||
return expr;
|
||||
}
|
||||
|
||||
public Expression modifyIntermediateField(IntermediateFieldExpression expr) {
|
||||
return expr;
|
||||
}
|
||||
|
||||
public Expression modifyNull(NullExpression expr) {
|
||||
return expr;
|
||||
}
|
||||
|
||||
@@ -4,6 +4,7 @@ import jef.expressions.BinaryExpression;
|
||||
import jef.expressions.ConstantExpression;
|
||||
import jef.expressions.Expression;
|
||||
import jef.expressions.FieldExpression;
|
||||
import jef.expressions.IntermediateFieldExpression;
|
||||
import jef.expressions.UnaryExpression;
|
||||
|
||||
import java.util.Set;
|
||||
@@ -18,6 +19,10 @@ public class IConst0Fixer extends ExpressionModifier {
|
||||
&& expr.getExpr().getType() == Expression.Type.FIELD
|
||||
&& DESCRIPTORS.contains(((FieldExpression) expr.getExpr()).getClassDescriptor())) {
|
||||
return modify(new BinaryExpression(expr.getExpr(), ConstantExpression.V0, BinaryExpression.Operator.EQ));
|
||||
} else if (expr.getOperator() == UnaryExpression.Operator.NOT
|
||||
&& expr.getExpr().getType() == Expression.Type.INTERMEDIATE_FIELD
|
||||
&& DESCRIPTORS.contains(((IntermediateFieldExpression) expr.getExpr()).getClassDescriptor())) {
|
||||
return modify(new BinaryExpression(expr.getExpr(), ConstantExpression.V0, BinaryExpression.Operator.EQ));
|
||||
}
|
||||
return super.modifyUnary(expr);
|
||||
}
|
||||
|
||||
@@ -2,18 +2,17 @@ package jef.expressions.modifier;
|
||||
|
||||
import jef.expressions.Expression;
|
||||
import jef.expressions.FieldExpression;
|
||||
import jef.expressions.IntermediateFieldExpression;
|
||||
|
||||
public class TableAliasInjector extends ExpressionModifier {
|
||||
private final String tableAlias;
|
||||
private final String prefix;
|
||||
|
||||
public TableAliasInjector(String tableAlias) {
|
||||
this.tableAlias = tableAlias;
|
||||
this.prefix = (tableAlias == null || tableAlias.isBlank()) ? "" : tableAlias + ".";
|
||||
}
|
||||
|
||||
@Override
|
||||
public Expression modifyField(FieldExpression expr) {
|
||||
return new FieldExpression(prefix + expr.getName(), expr.getClassDescriptor());
|
||||
public Expression modifyIntermediateField(IntermediateFieldExpression expr) {
|
||||
return new FieldExpression(null, tableAlias, expr.getName(), expr.getClassDescriptor());
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,25 @@
|
||||
package jef.expressions.selectable;
|
||||
|
||||
import jef.expressions.Expression;
|
||||
|
||||
public class DatabaseFunctionExpression implements Expression, SelectableExpression {
|
||||
private final String function;
|
||||
public DatabaseFunctionExpression(String function) {
|
||||
this.function = function;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Type getType() {
|
||||
return Type.CONSTANT;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Priority getPriority() {
|
||||
return Priority.UNDEFINED;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return function;
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,20 @@
|
||||
package jef.expressions.selectable;
|
||||
|
||||
import jef.expressions.Expression;
|
||||
|
||||
public class DatabaseSelectAllExpression implements Expression, SelectableExpression {
|
||||
@Override
|
||||
public Expression.Type getType() {
|
||||
return Expression.Type.CONSTANT;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Expression.Priority getPriority() {
|
||||
return Expression.Priority.UNDEFINED;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "*";
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,5 @@
|
||||
package jef.expressions.selectable;
|
||||
|
||||
public interface SelectableExpression {
|
||||
|
||||
}
|
||||
@@ -4,6 +4,7 @@ import jef.expressions.AndExpression;
|
||||
import jef.expressions.BinaryExpression;
|
||||
import jef.expressions.ConstantExpression;
|
||||
import jef.expressions.FieldExpression;
|
||||
import jef.expressions.IntermediateFieldExpression;
|
||||
import jef.expressions.NullExpression;
|
||||
import jef.expressions.OrExpression;
|
||||
import jef.expressions.ParameterExpression;
|
||||
@@ -12,6 +13,7 @@ import jef.expressions.TableExpression;
|
||||
import jef.expressions.TernaryExpression;
|
||||
import jef.expressions.UnaryExpression;
|
||||
import jef.expressions.WhereExpression;
|
||||
import jef.expressions.selectable.SelectableExpression;
|
||||
|
||||
import java.util.Collection;
|
||||
import java.util.stream.Collectors;
|
||||
@@ -48,7 +50,12 @@ public class DebugExpressionVisitor extends ExpressionVisitor {
|
||||
|
||||
@Override
|
||||
public void visitField(FieldExpression expr) {
|
||||
System.out.println(i() + expr.getName());
|
||||
System.out.println(i() + expr.toString());
|
||||
}
|
||||
|
||||
@Override
|
||||
public void visitIntermediateField(IntermediateFieldExpression expr) {
|
||||
System.out.println(i() + expr.toString());
|
||||
}
|
||||
|
||||
@Override
|
||||
@@ -81,7 +88,7 @@ public class DebugExpressionVisitor extends ExpressionVisitor {
|
||||
var table = expr.getFrom() instanceof TableExpression;
|
||||
var tableName = table ? ((TableExpression) expr.getFrom()).getName() : null;
|
||||
var tableAlias = (expr.getFromAlias() == null || expr.getFromAlias().isBlank()) ? "" : " " + expr.getFromAlias();
|
||||
System.out.println(i() + "SELECT " + expr.getFields().stream().collect(Collectors.joining(", "))
|
||||
System.out.println(i() + "SELECT " + expr.getFields().stream().map(SelectableExpression::toString).collect(Collectors.joining(", "))
|
||||
+ " FROM" + (!table ? " (" : tableName + " " + tableAlias));
|
||||
if (!table) {
|
||||
indent++;
|
||||
|
||||
@@ -5,6 +5,7 @@ import jef.expressions.BinaryExpression;
|
||||
import jef.expressions.ConstantExpression;
|
||||
import jef.expressions.Expression;
|
||||
import jef.expressions.FieldExpression;
|
||||
import jef.expressions.IntermediateFieldExpression;
|
||||
import jef.expressions.NullExpression;
|
||||
import jef.expressions.OrExpression;
|
||||
import jef.expressions.ParameterExpression;
|
||||
@@ -21,6 +22,7 @@ public abstract class ExpressionVisitor {
|
||||
case BINARY -> visitBinary((BinaryExpression) expr);
|
||||
case CONSTANT -> visitConstant((ConstantExpression) expr);
|
||||
case FIELD -> visitField((FieldExpression) expr);
|
||||
case INTERMEDIATE_FIELD -> visitIntermediateField((IntermediateFieldExpression) expr);
|
||||
case NULL -> visitNull((NullExpression) expr);
|
||||
case OR -> visitOr((OrExpression) expr);
|
||||
case PARAMETER -> visitParameter((ParameterExpression) expr);
|
||||
@@ -50,6 +52,9 @@ public abstract class ExpressionVisitor {
|
||||
public void visitField(FieldExpression expr) {
|
||||
}
|
||||
|
||||
public void visitIntermediateField(IntermediateFieldExpression expr) {
|
||||
}
|
||||
|
||||
public void visitNull(NullExpression expr) {
|
||||
}
|
||||
|
||||
|
||||
@@ -10,6 +10,7 @@ import jef.expressions.modifier.ExpressionOptimizerBottomUp;
|
||||
import jef.expressions.modifier.IConst0Fixer;
|
||||
import jef.expressions.modifier.TableAliasInjector;
|
||||
import jef.expressions.modifier.TernaryRewriter;
|
||||
import jef.expressions.selectable.DatabaseSelectAllExpression;
|
||||
import jef.serializable.SerializablePredicate;
|
||||
|
||||
import java.io.Serializable;
|
||||
@@ -50,7 +51,7 @@ public class FilterOp<T extends Serializable> implements Queryable<T>, Operation
|
||||
|
||||
@Override
|
||||
public Expression getExpression() {
|
||||
return new WhereExpression(new SelectExpression(List.of("*"), queryable.getExpression(), getTableAlias()), predicateExpr);
|
||||
return new WhereExpression(new SelectExpression(List.of(new DatabaseSelectAllExpression()), queryable.getExpression(), getTableAlias()), predicateExpr);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
||||
Reference in New Issue
Block a user