io.odysz.transact.sql.parts.condition.Predicate Maven / Gradle / Ivy
package io.odysz.transact.sql.parts.condition;
import static io.odysz.transact.sql.parts.Logic.op.eq;
import static io.odysz.transact.sql.parts.Logic.op.isnull;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import io.odysz.semantics.ISemantext;
import io.odysz.transact.sql.Query;
import io.odysz.transact.sql.parts.AbsPart;
import io.odysz.transact.sql.parts.Logic;
import io.odysz.transact.sql.parts.antlr.ExprsVisitor;
import io.odysz.transact.x.TransException;
/**Basically a logic expression. For predicate definition, see {@link io.odysz.transact.sql.parts.antlr.PredicatVisitor}.
* @author [email protected]
*/
public class Predicate extends AbsPart {
protected boolean empty = false;
public boolean empty() { return empty; }
// private boolean negative = false;
// public boolean negative() { return negative; }
private Logic.op op;
public Logic.op logic() { return op; }
private ExprPart l;
private ExprPart r;
private boolean brace;
private Condit search_condit;
private Query inSelect;
public Predicate(Logic.op op, ExprPart lexpr, String rop) {
this.op = op;
this.l = lexpr;
// this.r = new ExprPart(rop);
this.r = rop == null ? null : ExprsVisitor.parse(rop);
}
public Predicate() {
}
public Predicate(Predicate cpy) {
if (cpy == null)
empty = true;
else {
empty = false;
this.op = cpy.op;
this.l = cpy.l;
this.r = cpy.r;
this.brace = cpy.brace;
this.search_condit = cpy.search_condit;
this.inSelect = cpy.inSelect;
}
}
public Predicate(Logic.op op, ExprPart lexpr, ExprPart rexpr) {
this.op = op;
this.l = lexpr;
this.r = rexpr;
}
public Predicate(Logic.op inlike, ExprPart expression,
List inlikes, boolean... not) {
op = inlike;
l = expression;
String rstr = inlikes.stream()
.map(var -> {
try {
return var.sql(null);
} catch (TransException e) {
e.printStackTrace();
return e.getMessage();
}
}) // variable resolving goes here
.collect(Collectors.joining(", "));
r = new ExprPart(rstr);
}
public Predicate(Logic.op op, String lp, String rp) {
this.op = op;
this.l = lp == null ? null : ExprsVisitor.parse(lp);
this.r = rp == null ? null :
// for IN's rp, it's an array of expressions, ExprsVisitor can't parse it
op == Logic.op.in ? new ExprPart(rp)
: ExprsVisitor.parse(rp);
}
public Predicate(Logic.op op, String lp, ExprPart rp) {
this.op = op;
// this.l = new ExprPart(from);
this.l = lp == null ? null : ExprsVisitor.parse(lp);
this.r = rp;
}
/**Use this to visit predicate : '(' search_condition ')';
* @param search_condit
*/
public Predicate(Condit search_condit) {
this.brace = true;
this.search_condit = search_condit;
}
public Predicate(Logic.op in, String col, Query q) throws TransException {
this.op = in;
/* 2024-03-19, it's valid for where clause like where 1 >= (select (count(*) form ...))
if (in != Logic.op.in && in != Logic.op.notin)
throw new TransException("Currently only '(not) in' operator is supported for select condition. select:\n %s", s.sql(null));
*/
// this.l = new ExprPart(col);
this.l = col == null ? null : ExprsVisitor.parse(col);
this.inSelect = q;
}
// 2024-03-19
// public void not(TerminalNode not) {
// negative = not != null && not.getText() != null && not.getText().length() > 0;
// }
@Override
public String sql(ISemantext sctx) throws TransException {
if (empty) return "";
// else if ((op == Logic.op.in || op == Logic.op.notin) && inSelect != null)
else if (inSelect != null) {
return this.l == null && r != null
? Stream.of(new ExprPart("("), inSelect, new ExprPart(")"), new ExprPart(op.sql(sctx, op, r)))
.map(e -> {
try {
return e == null ? "" : e.sql(sctx);
} catch (TransException e1) {
e1.printStackTrace();
return "";
}
})
.collect(Collectors.joining(" "))
: this.l != null
? Stream.of(l, new ExprPart(op.sql(sctx, op, new ExprPart(""))), new ExprPart("("), inSelect, new ExprPart(")"))
.map(e -> {
try {
return e == null ? "" : e.sql(sctx);
} catch (TransException e1) {
e1.printStackTrace();
return "";
}
})
.collect(Collectors.joining(" "))
: Stream.of(new ExprPart(op.sql(sctx, op, new ExprPart(""))), new ExprPart("("), inSelect, new ExprPart(")"))
.map( (e) -> {
try { return e.sql(sctx); }
catch (TransException e1) {
e1.printStackTrace();
return op.sql(sctx, op, "");
}})
.collect(Collectors.joining(" "));
// inSelect.sql(sctx);
}
if (brace && search_condit != null)
return String.format("(%s)", search_condit.sql(sctx));
else {
// return String.format("%s %s", l.sql(sctx), op.sql(sctx, op, r.sql(sctx), negative));
// FIXME Should pedicate's operands always can't escape ' ?
if (l == null) throw new TransException(
"Predictat sql can't built with null left operand. op: %s", op.name());
return String.format("%s %s", l.escape(escape).sql(sctx),
op == eq && r == null
? isnull.sql(sctx, isnull, null)
: op.sql(sctx, op, r == null ? "" : r.escape(escape).sql(sctx)));
}
}
/** escape ' with ''
* Predicate's default escape is false?*/
boolean escape = false;
/** enable escape - replace ' with ''. (default true)
* @param esc
* @return this
*/
public Predicate escape(boolean esc) {
escape = esc;
return this;
}
/**Helper for creating a equal predicate.
* @param col
* @param constv
* @return predicate for col = 'constv'
*/
public static Predicate eq(String col, String constv) {
return new Predicate(Logic.op.eq, col, "'" + constv + "'");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy