All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.redhat.lightblue.assoc.qrew.QueryRewriter Maven / Gradle / Ivy

There is a newer version: 2.18.0
Show newest version
/*
 Copyright 2013 Red Hat, Inc. and/or its affiliates.

 This file is part of lightblue.

 This program is free software: you can redistribute it and/or modify
 it under the terms of the GNU General Public License as published by
 the Free Software Foundation, either version 3 of the License, or
 (at your option) any later version.

 This program is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 GNU General Public License for more details.

 You should have received a copy of the GNU General Public License
 along with this program.  If not, see .
 */
package com.redhat.lightblue.assoc.qrew;

import java.util.Set;
import java.util.HashSet;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.redhat.lightblue.query.QueryExpression;
import com.redhat.lightblue.query.UnaryLogicalExpression;
import com.redhat.lightblue.query.NaryLogicalExpression;
import com.redhat.lightblue.query.ArrayMatchExpression;

import com.redhat.lightblue.util.CopyOnWriteIterator;

import com.redhat.lightblue.assoc.qrew.rules.*;

/**
 * Implementation of Rewriter that orchestrates rewriting rules with registered Rewriter instances.
 */
public final class QueryRewriter extends Rewriter {

    private static final Logger LOGGER=LoggerFactory.getLogger(QueryRewriter.class);

    private static final Set rewriteRules=new HashSet<>(16);

    public QueryRewriter() {
        this(true);
    }

    public QueryRewriter(boolean withDefaultRules) {
        if(withDefaultRules)
            registerDefaultRules();
    }

    /**
     * Register a Rewriter instance.  If attempt to register QueryRewriter returns without doing anything.
     * @param rule the rewriter to register
     */
    public void register(Rewriter rule) {
        if (rule instanceof QueryRewriter) {
            return;
        }
        rewriteRules.add(rule);
    }

    /**
     * Register default rules.
     */
    protected void registerDefaultRules() {
        register(CombineANDsToNIN.INSTANCE);
        register(CombineINsInOR.INSTANCE);
        register(CombineNINsInAND.INSTANCE);
        register(CombineORsToIN.INSTANCE);
        register(EliminateNOT.INSTANCE);
        register(EliminateNOTNOT.INSTANCE);
        register(EliminateNOTOR.INSTANCE);
        register(EliminateSingleANDOR.INSTANCE);
        register(ExtendINsInOR.INSTANCE);
        register(ExtendNINsInAND.INSTANCE);
        register(PromoteNestedAND.INSTANCE);
        register(SimpleElemMatchIsComparison.INSTANCE);
    }

    protected QueryExpression rewriteIteration(QueryExpression q) {
        LOGGER.debug("Rewrite iteration begins for q={}",q);
        QueryExpression newq=q;
        if (q instanceof UnaryLogicalExpression) {
            LOGGER.debug("q is a unary logical expression, rewriting nested query");
            QueryExpression nestedq=((UnaryLogicalExpression)q).getQuery();
            QueryExpression newNestedq=rewriteIteration(nestedq);
            LOGGER.debug("Rewritten nested query={}",newNestedq);
            if(newNestedq!=nestedq)
                newq=new UnaryLogicalExpression( ((UnaryLogicalExpression)q).getOp(), newNestedq);
        } else if (q instanceof NaryLogicalExpression) {
            LOGGER.debug("q is a n-ary logical expression, rewriting nested terms");
            CopyOnWriteIterator cowr=new CopyOnWriteIterator<>( ((NaryLogicalExpression)q).getQueries());
            while(cowr.hasNext()) {
                QueryExpression nestedq=cowr.next();
                QueryExpression newNestedq=rewriteIteration(nestedq);
                if(newNestedq!=nestedq)
                    cowr.set(newNestedq);
            }
            if(cowr.isCopied())
                newq=new NaryLogicalExpression( ((NaryLogicalExpression)q).getOp(),cowr.getCopiedList());
        } else if (q instanceof ArrayMatchExpression) {
            LOGGER.debug("q is an array match expression, rewriting nested query");
            QueryExpression nestedq=((ArrayMatchExpression)q).getElemMatch();
            QueryExpression newNestedq=rewriteIteration(nestedq);
            LOGGER.debug("Rewritten nested query={}",newNestedq);
            if(newNestedq!=nestedq)
                newq=new ArrayMatchExpression( ((ArrayMatchExpression)q).getArray(),newNestedq);
        } 
        LOGGER.debug("Applying rewrite rules to q");
        newq=applyRules(newq);
        LOGGER.debug("Rewritten q={}",newq);
        
        return newq;
    }

    @Override
    public QueryExpression rewrite(QueryExpression q) {
        QueryExpression trc=q;
        QueryExpression newq;
        boolean done=false;
        do {
            newq=rewriteIteration(trc);
            LOGGER.debug("Rewrite iteration pre={}",trc);
            LOGGER.debug("Rewrite iteration post={}",newq);
            if(newq==trc)
                done=true;
            trc=newq;
        } while(!done);
        return newq;
    }

    private QueryExpression applyRules(QueryExpression q) {
        QueryExpression newq=q;
        for(Rewriter r:rewriteRules)
            newq=r.rewrite(newq);
        return newq;
    }
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy