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

org.apache.openjpa.jdbc.kernel.exps.EndsWithExpression Maven / Gradle / Ivy

There is a newer version: 4.0.1
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.apache.openjpa.jdbc.kernel.exps;

import java.util.Map;

import org.apache.openjpa.jdbc.meta.ClassMapping;
import org.apache.openjpa.jdbc.meta.FieldMapping;
import org.apache.openjpa.jdbc.schema.Column;
import org.apache.openjpa.jdbc.schema.Table;
import org.apache.openjpa.jdbc.sql.DBDictionary;
import org.apache.openjpa.jdbc.sql.SQLBuffer;
import org.apache.openjpa.jdbc.sql.Select;
import org.apache.openjpa.kernel.exps.ExpressionVisitor;
import org.apache.openjpa.meta.XMLMetaData;

/**
 * Test if one string ends with another.
 *
 * @author Abe White
 */
class EndsWithExpression
    implements Exp {

    
    private static final long serialVersionUID = 1L;
    private final Val _val1;
    private final Val _val2;

    /**
     * Constructor. Supply values.
     */
    public EndsWithExpression(Val val1, Val val2) {
        _val1 = val1;
        _val2 = val2;
    }

    @Override
    public ExpState initialize(Select sel, ExpContext ctx, Map contains) {
        ExpState s1 = _val1.initialize(sel, ctx, 0);
        ExpState s2 = _val2.initialize(sel, ctx, 0);
        return new BinaryOpExpState(sel.and(s1.joins, s2.joins), s1, s2);
    }

    @Override
    public void appendTo(Select sel, ExpContext ctx, ExpState state,
        SQLBuffer buf) {
        BinaryOpExpState bstate = (BinaryOpExpState) state;
        _val1.calculateValue(sel, ctx, bstate.state1, _val2, bstate.state2);
        _val2.calculateValue(sel, ctx, bstate.state2, _val1, bstate.state1);

        DBDictionary dict = ctx.store.getDBDictionary();
        String func = dict.stringLengthFunction;
        String pre = null;
        String post = null;
        if (func != null) {
            int idx = func.indexOf("{0}");
            pre = func.substring(0, idx);
            post = func.substring(idx + 3);
        }

        if (_val1 instanceof Const && ((Const) _val1).getValue(ctx,
            bstate.state1) == null)
            buf.append("1 <> 1");
        else if (_val2 instanceof Const) {
            Object o = ((Const) _val2).getValue(ctx, bstate.state2);
            if (o == null)
                buf.append("1 <> 1");
            else {
                Column col = null;
                if (_val1 instanceof PCPath) {
                    Column[] cols = ((PCPath) _val1).getColumns(bstate.state1);
                    if (cols.length == 1)
                        col = cols[0];
                }

                _val1.appendTo(sel, ctx, bstate.state1, buf, 0);
                buf.append(" LIKE ");
                buf.appendValue("%" + o.toString(), col);
            }
        } else {
            // if we can't use LIKE, we have to take the substring of the
            // first value and compare it to the second
            dict.assertSupport(pre != null, "StringLengthFunction");
            dict.substring(buf,
                new FilterValueImpl(sel, ctx, bstate.state1, _val1),
                new StringLengthDifferenceFilterValue(sel, ctx, bstate, pre,
                    post), null);
            buf.append(" = ");
            _val2.appendTo(sel, ctx, bstate.state2, buf, 0);
        }

        sel.append(buf, state.joins);
    }

    @Override
    public void selectColumns(Select sel, ExpContext ctx, ExpState state,
        boolean pks) {
        BinaryOpExpState bstate = (BinaryOpExpState) state;
        _val1.selectColumns(sel, ctx, bstate.state1, true);
        _val2.selectColumns(sel, ctx, bstate.state2, true);
    }

    @Override
    public void acceptVisit(ExpressionVisitor visitor) {
        visitor.enter(this);
        _val1.acceptVisit(visitor);
        _val2.acceptVisit(visitor);
        visitor.exit(this);
    }

    /**
     * Evaluates to the length of a given value.
     */
    private class StringLengthDifferenceFilterValue
        implements FilterValue {

        private final Select _sel;
        private final ExpContext _ctx;
        private final BinaryOpExpState _state;
        private final String _pre;
        private final String _post;

        public StringLengthDifferenceFilterValue(Select sel, ExpContext ctx,
            BinaryOpExpState state, String pre, String post) {
            _sel = sel;
            _ctx = ctx;
            _state = state;
            _pre = pre;
            _post = post;
        }

        @Override
        public Class getType() {
            return int.class;
        }

        @Override
        public int length() {
            return 1;
        }

        @Override
        public void appendTo(SQLBuffer buf) {
            appendTo(buf, 0);
        }

        @Override
        public void appendTo(SQLBuffer buf, int index) {
            buf.append(_pre);
            _val1.appendTo(_sel, _ctx, _state.state1, buf, index);
            buf.append(_post).append(" - ").append(_pre);
            _val2.appendTo(_sel, _ctx, _state.state2, buf, index);
            buf.append(_post);
        }

        @Override
        public String getColumnAlias(Column col) {
            return _sel.getColumnAlias(col, _state.joins);
        }

        @Override
        public String getColumnAlias(String col, Table table) {
            return _sel.getColumnAlias(col, table, _state.joins);
        }

        @Override
        public Object toDataStoreValue(Object val) {
            return val;
        }

        @Override
        public boolean isConstant() {
            return false;
        }

        @Override
        public Object getValue() {
            return null;
        }

        @Override
        public Object getSQLValue() {
            return null;
        }

        @Override
        public boolean isPath() {
            return false;
        }

        @Override
        public ClassMapping getClassMapping() {
            return null;
        }

        @Override
        public FieldMapping getFieldMapping() {
            return null;
        }

        @Override
        public PCPath getXPath() {
            return null;
        }

        @Override
        public XMLMetaData getXmlMapping() {
            return null;
        }

        @Override
        public boolean requiresCast() {
            return false;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy