blob: c5c63923b9113672a746b001255867649a7e7360 [file] [log] [blame]
/*
* 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.jackrabbit.oak.query.ast;
import java.util.List;
import java.util.Set;
import javax.jcr.PropertyType;
import org.apache.jackrabbit.oak.api.PropertyValue;
import org.apache.jackrabbit.oak.api.Type;
import org.apache.jackrabbit.oak.query.index.FilterImpl;
import org.apache.jackrabbit.oak.plugins.memory.PropertyValues;
import org.apache.jackrabbit.oak.spi.query.QueryConstants;
import org.apache.jackrabbit.oak.spi.query.QueryIndex.OrderEntry;
/**
* The function "length(..)".
*/
public class LengthImpl extends DynamicOperandImpl {
private final DynamicOperandImpl operand;
public LengthImpl(DynamicOperandImpl operand) {
this.operand = operand;
}
public DynamicOperandImpl getOperand() {
return operand;
}
@Override
boolean accept(AstVisitor v) {
return v.visit(this);
}
@Override
public String toString() {
return "length(" + operand + ')';
}
@Override
public PropertyExistenceImpl getPropertyExistence() {
return operand.getPropertyExistence();
}
@Override
public Set<SelectorImpl> getSelectors() {
return operand.getSelectors();
}
@Override
public PropertyValue currentProperty() {
PropertyValue p = operand.currentProperty();
if (p == null) {
return null;
}
// TODO namespace remapping?
if (!p.isArray()) {
long length = p.size();
return PropertyValues.newLong(length);
}
// TODO what is the expected result for LENGTH(multiValueProperty)?
throw new IllegalArgumentException("LENGTH(x) on multi-valued property is not supported");
}
@Override
public void restrict(FilterImpl f, Operator operator, PropertyValue v) {
if (v != null) {
switch (v.getType().tag()) {
case PropertyType.LONG:
case PropertyType.DECIMAL:
case PropertyType.DOUBLE:
// ok - comparison with a number
break;
case PropertyType.BINARY:
case PropertyType.STRING:
case PropertyType.DATE:
// ok - compare with a string literal
break;
default:
throw new IllegalArgumentException(
"Can not compare the length with a constant of type "
+ PropertyType.nameFromValue(v.getType().tag()) +
" and value " + v.toString());
}
}
// LENGTH(x) implies x is not null
operand.restrict(f, Operator.NOT_EQUAL, null);
if (operator == Operator.NOT_EQUAL && v != null) {
// not supported
return;
}
String fn = getFunction(f.getSelector());
if (fn != null) {
f.restrictProperty(QueryConstants.FUNCTION_RESTRICTION_PREFIX + fn,
operator, v, PropertyType.LONG);
}
}
@Override
public void restrictList(FilterImpl f, List<PropertyValue> list) {
// optimizations of the type "length(x) in(1, 2, 3)" are not supported
}
@Override
public String getFunction(SelectorImpl s) {
String f = operand.getFunction(s);
if (f == null) {
return null;
}
return "length*" + f;
}
@Override
public boolean canRestrictSelector(SelectorImpl s) {
return operand.canRestrictSelector(s);
}
@Override
int getPropertyType() {
return PropertyType.LONG;
}
@Override
public DynamicOperandImpl createCopy() {
return new LengthImpl(operand.createCopy());
}
@Override
public OrderEntry getOrderEntry(SelectorImpl s, OrderingImpl o) {
String fn = getFunction(s);
if (fn != null) {
return new OrderEntry(
QueryConstants.FUNCTION_RESTRICTION_PREFIX + fn,
Type.LONG,
o.isDescending() ?
OrderEntry.Order.DESCENDING : OrderEntry.Order.ASCENDING);
}
return null;
}
}