| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251 |
- package org.elasticsearch.dsl.parser;
- import com.alibaba.druid.sql.ast.SQLExpr;
- import com.alibaba.druid.sql.ast.expr.*;
- import com.google.common.collect.Lists;
- import org.apache.commons.collections.CollectionUtils;
- import org.elasticsearch.dsl.ElasticDslContext;
- import org.elasticsearch.dsl.ElasticSqlParseUtil;
- import org.elasticsearch.dsl.exception.ElasticSql2DslException;
- import org.elasticsearch.dsl.parser.helper.ElasticSqlIdentifierHelper;
- import org.elasticsearch.index.query.BoolFilterBuilder;
- import org.elasticsearch.index.query.FilterBuilder;
- import org.elasticsearch.index.query.FilterBuilders;
- import org.elasticsearch.sql.ElasticSqlSelectQueryBlock;
- import java.util.List;
- import java.util.function.Consumer;
- public class QueryWhereConditionParser implements QueryParser {
- @Override
- public void parse(ElasticDslContext dslContext) {
- ElasticSqlSelectQueryBlock queryBlock = (ElasticSqlSelectQueryBlock) dslContext.getQueryExpr().getSubQuery().getQuery();
- if (queryBlock.getWhere() != null) {
- SqlCondition sqlCondition = parseFilterCondition(dslContext, queryBlock.getWhere());
- if (!sqlCondition.isAndOr()) {
- dslContext.getParseResult().boolFilter().must(sqlCondition.getFilterList().get(0));
- } else {
- for (FilterBuilder filter : sqlCondition.getFilterList()) {
- if (sqlCondition.getOperator() == SQLBinaryOperator.BooleanAnd) {
- dslContext.getParseResult().boolFilter().must(filter);
- }
- if (sqlCondition.getOperator() == SQLBinaryOperator.BooleanOr) {
- dslContext.getParseResult().boolFilter().should(filter);
- }
- }
- }
- }
- }
- private SqlCondition parseFilterCondition(ElasticDslContext dslContext, SQLExpr sqlExpr) {
- if (sqlExpr instanceof SQLBinaryOpExpr) {
- SQLBinaryOpExpr sqlBinOpExpr = (SQLBinaryOpExpr) sqlExpr;
- SQLBinaryOperator binaryOperator = sqlBinOpExpr.getOperator();
- if (SQLBinaryOperator.BooleanAnd == binaryOperator || SQLBinaryOperator.BooleanOr == binaryOperator) {
- final SqlCondition leftCondition = parseFilterCondition(dslContext, sqlBinOpExpr.getLeft());
- final SqlCondition rightCondition = parseFilterCondition(dslContext, sqlBinOpExpr.getRight());
- List<FilterBuilder> curFilterList = Lists.newArrayList();
- if (!leftCondition.isAndOr() || leftCondition.operator == binaryOperator) {
- curFilterList.addAll(leftCondition.getFilterList());
- } else {
- final BoolFilterBuilder subBoolFilter = FilterBuilders.boolFilter();
- leftCondition.getFilterList().stream().forEach(new Consumer<FilterBuilder>() {
- @Override
- public void accept(FilterBuilder filter) {
- if (leftCondition.getOperator() == SQLBinaryOperator.BooleanAnd) {
- subBoolFilter.must(filter);
- }
- if (leftCondition.getOperator() == SQLBinaryOperator.BooleanOr) {
- subBoolFilter.should(filter);
- }
- }
- });
- curFilterList.add(subBoolFilter);
- }
- if (!rightCondition.isAndOr() || rightCondition.operator == binaryOperator) {
- curFilterList.addAll(rightCondition.getFilterList());
- } else {
- final BoolFilterBuilder subBoolFilter = FilterBuilders.boolFilter();
- rightCondition.getFilterList().stream().forEach(new Consumer<FilterBuilder>() {
- @Override
- public void accept(FilterBuilder filter) {
- if (rightCondition.getOperator() == SQLBinaryOperator.BooleanAnd) {
- subBoolFilter.must(filter);
- }
- if (rightCondition.getOperator() == SQLBinaryOperator.BooleanOr) {
- subBoolFilter.should(filter);
- }
- }
- });
- curFilterList.add(subBoolFilter);
- }
- return new SqlCondition(curFilterList, binaryOperator);
- }
- }
- return new SqlCondition(parseAtomFilterCondition(dslContext, sqlExpr));
- }
- private FilterBuilder parseAtomFilterCondition(ElasticDslContext dslContext, SQLExpr sqlExpr) {
- if (sqlExpr instanceof SQLBinaryOpExpr) {
- SQLBinaryOpExpr sqlBinOpExpr = (SQLBinaryOpExpr) sqlExpr;
- final SQLBinaryOperator binaryOperator = sqlBinOpExpr.getOperator();
- if (ElasticSqlParseUtil.isValidBinOperator(binaryOperator)) {
- //以下是二元运算,左边必须是变量
- if (!(sqlBinOpExpr.getLeft() instanceof SQLPropertyExpr || sqlBinOpExpr.getLeft() instanceof SQLIdentifierExpr)) {
- throw new ElasticSql2DslException("[syntax error] Binary operation expr left part should be a param name");
- }
- //EQ NEQ
- if (SQLBinaryOperator.Equality == binaryOperator || SQLBinaryOperator.LessThanOrGreater == binaryOperator || SQLBinaryOperator.NotEqual == binaryOperator) {
- final Object targetVal = ElasticSqlParseUtil.transferSqlArg(sqlBinOpExpr.getRight(), dslContext.getSqlArgs());
- return parseCondition(sqlBinOpExpr.getLeft(), dslContext.getParseResult().getQueryAs(), new ConditionFilterBuilder() {
- @Override
- public FilterBuilder buildFilter(String idfName) {
- FilterBuilder eqFilter = FilterBuilders.termFilter(idfName, targetVal);
- if (SQLBinaryOperator.Equality == binaryOperator) {
- return eqFilter;
- } else {
- return FilterBuilders.notFilter(eqFilter);
- }
- }
- });
- }
- //GT GTE LT LTE
- if (SQLBinaryOperator.GreaterThan == binaryOperator || SQLBinaryOperator.GreaterThanOrEqual == binaryOperator
- || SQLBinaryOperator.LessThan == binaryOperator || SQLBinaryOperator.LessThanOrEqual == binaryOperator) {
- final Object targetVal = ElasticSqlParseUtil.transferSqlArg(sqlBinOpExpr.getRight(), dslContext.getSqlArgs());
- return parseCondition(sqlBinOpExpr.getLeft(), dslContext.getParseResult().getQueryAs(), new ConditionFilterBuilder() {
- @Override
- public FilterBuilder buildFilter(String idfName) {
- FilterBuilder rangeFilter = null;
- if (SQLBinaryOperator.GreaterThan == binaryOperator) {
- rangeFilter = FilterBuilders.rangeFilter(idfName).gt(targetVal);
- } else if (SQLBinaryOperator.GreaterThanOrEqual == binaryOperator) {
- rangeFilter = FilterBuilders.rangeFilter(idfName).gte(targetVal);
- } else if (SQLBinaryOperator.LessThan == binaryOperator) {
- rangeFilter = FilterBuilders.rangeFilter(idfName).lt(targetVal);
- } else if (SQLBinaryOperator.LessThanOrEqual == binaryOperator) {
- rangeFilter = FilterBuilders.rangeFilter(idfName).lte(targetVal);
- }
- return rangeFilter;
- }
- });
- }
- //IS / IS NOT
- if (SQLBinaryOperator.Is == binaryOperator || SQLBinaryOperator.IsNot == binaryOperator) {
- if (!(sqlBinOpExpr.getRight() instanceof SQLNullExpr)) {
- throw new ElasticSql2DslException("[syntax error] Is/IsNot expr right part should be null");
- }
- return parseCondition(sqlBinOpExpr.getLeft(), dslContext.getParseResult().getQueryAs(), new ConditionFilterBuilder() {
- @Override
- public FilterBuilder buildFilter(String idfName) {
- FilterBuilder missingFilter = FilterBuilders.missingFilter(idfName);
- if (SQLBinaryOperator.IsNot == binaryOperator) {
- return FilterBuilders.notFilter(missingFilter);
- }
- return missingFilter;
- }
- });
- }
- }
- } else if (sqlExpr instanceof SQLInListExpr) {
- final SQLInListExpr inListExpr = (SQLInListExpr) sqlExpr;
- if (CollectionUtils.isEmpty(inListExpr.getTargetList())) {
- throw new ElasticSql2DslException("[syntax error] In list expr target list cannot be blank");
- }
- final Object[] targetInList = ElasticSqlParseUtil.transferSqlArgs(inListExpr.getTargetList(), dslContext.getSqlArgs());
- return parseCondition(inListExpr.getExpr(), dslContext.getParseResult().getQueryAs(), new ConditionFilterBuilder() {
- @Override
- public FilterBuilder buildFilter(String idfName) {
- if (inListExpr.isNot()) {
- return FilterBuilders.notFilter(FilterBuilders.inFilter(idfName, targetInList));
- } else {
- return FilterBuilders.inFilter(idfName, targetInList);
- }
- }
- });
- } else if (sqlExpr instanceof SQLBetweenExpr) {
- SQLBetweenExpr betweenExpr = (SQLBetweenExpr) sqlExpr;
- final Object from = ElasticSqlParseUtil.transferSqlArg(betweenExpr.getBeginExpr(), dslContext.getSqlArgs());
- final Object to = ElasticSqlParseUtil.transferSqlArg(betweenExpr.getEndExpr(), dslContext.getSqlArgs());
- if (from == null || to == null) {
- throw new ElasticSql2DslException("[syntax error] Between Expr only support one of [number,date] arg type");
- }
- return parseCondition(betweenExpr.getTestExpr(), dslContext.getParseResult().getQueryAs(), new ConditionFilterBuilder() {
- @Override
- public FilterBuilder buildFilter(String idfName) {
- return FilterBuilders.rangeFilter(idfName).gte(from).lte(to);
- }
- });
- }
- throw new ElasticSql2DslException("[syntax error] Can not support syntax type: " + sqlExpr.toString());
- }
- private FilterBuilder parseCondition(SQLExpr sqlExpr, String queryAs, final ConditionFilterBuilder filterBuilder) {
- final List<FilterBuilder> tmpFilterList = Lists.newLinkedList();
- ElasticSqlIdentifierHelper.parseSqlIdentifier(sqlExpr, queryAs, new ElasticSqlIdentifierHelper.ElasticSqlTopIdfFunc() {
- @Override
- public void parse(String idfName) {
- FilterBuilder originalFilter = filterBuilder.buildFilter(idfName);
- tmpFilterList.add(originalFilter);
- }
- }, new ElasticSqlIdentifierHelper.ElasticSqlNestIdfFunc() {
- @Override
- public void parse(String nestPath, String idfName) {
- FilterBuilder originalFilter = filterBuilder.buildFilter(idfName);
- FilterBuilder nestFilter = FilterBuilders.nestedFilter(nestPath, originalFilter);
- tmpFilterList.add(nestFilter);
- }
- });
- if (CollectionUtils.isNotEmpty(tmpFilterList)) {
- return tmpFilterList.get(0);
- }
- return null;
- }
- @FunctionalInterface
- private interface ConditionFilterBuilder {
- FilterBuilder buildFilter(String idfName);
- }
- private class SqlCondition {
- //是否AND/OR运算
- private boolean isAndOr = false;
- //运算符
- private SQLBinaryOperator operator;
- //条件集合
- private List<FilterBuilder> filterList;
- public SqlCondition(FilterBuilder atomFilter) {
- filterList = Lists.newArrayList(atomFilter);
- isAndOr = false;
- }
- public SqlCondition(List<FilterBuilder> filterList, SQLBinaryOperator operator) {
- this.filterList = filterList;
- isAndOr = true;
- this.operator = operator;
- }
- public boolean isAndOr() {
- return isAndOr;
- }
- public SQLBinaryOperator getOperator() {
- return operator;
- }
- public List<FilterBuilder> getFilterList() {
- return filterList;
- }
- }
- }
|