[OLINGO-1358] Substringof Method support

This commit is contained in:
ramya vasanth 2019-05-14 14:51:23 +05:30
parent 24fed9b1b6
commit 9244d682f7
11 changed files with 529 additions and 3 deletions

View File

@ -1080,4 +1080,21 @@ public class FilterSystemQueryITCase extends AbstractParamTecSvcITCase {
assertEquals(errorCode.getStatusCode(), e.getStatusLine().getStatusCode());
}
}
@Test
public void substringOf() {
ODataRetrieveResponse<ClientEntitySet> response =
sendRequest(ES_ALL_PRIM, "substringof('Second',PropertyString)");
assertEquals(1, response.getBody().getEntities().size());
assertEquals("Second Resource - negative values",
response.getBody().getEntities().get(0).getProperty("PropertyString")
.getPrimitiveValue().toValue());
}
@Test
public void substringOfWithNegativeValues() {
fail(ES_ALL_PRIM, "substringof(123,PropertyString)", HttpStatusCode.BAD_REQUEST);
fail(ES_ALL_PRIM, "substringof(PropertyString,123)", HttpStatusCode.BAD_REQUEST);
fail(ES_ALL_PRIM, "substringof(123,123)", HttpStatusCode.BAD_REQUEST);
}
}

View File

@ -405,4 +405,26 @@ public class BasicHttpITCase extends AbstractBaseTestITCase {
assertEquals("3", IOUtils.toString(connection.getInputStream()));
connection.disconnect();
}
@Test
public void testSubstringOfWithParameterAlias() throws Exception {
URL url = new URL(SERVICE_URI + "ESAllPrim?$filter=substringof(@word,PropertyString)&@word='Second'");
HttpURLConnection connection = (HttpURLConnection) url.openConnection();
connection.setRequestMethod(HttpMethod.GET.name());
connection.setRequestProperty(HttpHeader.ACCEPT, "application/json");
connection.connect();
assertEquals(HttpStatusCode.OK.getStatusCode(), connection.getResponseCode());
String content = IOUtils.toString(connection.getInputStream());
assertNotNull(content);
assertTrue(content.contains("\"value\":[{\"PropertyInt16\":-32768,"
+ "\"PropertyString\":\"Second Resource - negative values\","
+ "\"PropertyBoolean\":false,\"PropertyByte\":0,\"PropertySByte\":-128,\"PropertyInt32\":-2147483648,"
+ "\"PropertyInt64\":-9223372036854775808,\"PropertySingle\":-1.79E8,\"PropertyDouble\":-179000.0,"
+ "\"PropertyDecimal\":-34,\"PropertyBinary\":\"ASNFZ4mrze8=\",\"PropertyDate\":\"2015-11-05\","
+ "\"PropertyDateTimeOffset\":\"2005-12-03T07:17:08Z\",\"PropertyDuration\":\"PT9S\","
+ "\"PropertyGuid\":\"76543201-23ab-cdef-0123-456789dddfff\",\"PropertyTimeOfDay\":\"23:49:14\"}]}"));
connection.disconnect();
}
}

View File

@ -52,7 +52,8 @@ public enum MethodKind {
GEOLENGTH("geo.length"),
GEOINTERSECTS("geo.intersects"),
CAST("cast"),
ISOF("isof");
ISOF("isof"),
SUBSTRINGOF("substringof");
private String syntax;

View File

@ -153,6 +153,7 @@ public class ExpressionParser {
temp.put(TokenKind.ToupperMethod, MethodKind.TOUPPER);
temp.put(TokenKind.TrimMethod, MethodKind.TRIM);
temp.put(TokenKind.YearMethod, MethodKind.YEAR);
temp.put(TokenKind.SubstringofMethod, MethodKind.SUBSTRINGOF);
tokenToMethod = Collections.unmodifiableMap(temp);
}
@ -527,6 +528,7 @@ public class ExpressionParser {
case STARTSWITH:
case INDEXOF:
case CONCAT:
case SUBSTRINGOF:
ParserHelper.bws(tokenizer);
final Expression stringParameter1 = parseExpression();
checkType(stringParameter1, EdmPrimitiveTypeKind.String);

View File

@ -170,6 +170,7 @@ public class UriTokenizer {
ToupperMethod,
TrimMethod,
YearMethod,
SubstringofMethod,
IsDefinedMethod, // for the aggregation extension
@ -638,6 +639,9 @@ public class UriTokenizer {
case YearMethod:
found = nextMethod("year");
break;
case SubstringofMethod:
found = nextMethod("substringof");
break;
// Method for the aggregation extension
case IsDefinedMethod:

View File

@ -54,6 +54,7 @@ public class MethodImpl implements Method {
case CONTAINS:
case STARTSWITH:
case ENDSWITH:
case SUBSTRINGOF:
kind = EdmPrimitiveTypeKind.Boolean;
break;
case LENGTH:

View File

@ -25,14 +25,30 @@ import static org.junit.Assert.fail;
import static org.mockito.Mockito.mock;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.Locale;
import java.util.Map;
import org.apache.olingo.commons.api.edm.Edm;
import org.apache.olingo.commons.api.edm.EdmComplexType;
import org.apache.olingo.commons.api.edm.EdmEntityContainer;
import org.apache.olingo.commons.api.edm.EdmEntitySet;
import org.apache.olingo.commons.api.edm.EdmEntityType;
import org.apache.olingo.commons.api.edm.EdmKeyPropertyRef;
import org.apache.olingo.commons.api.edm.EdmNavigationProperty;
import org.apache.olingo.commons.api.edm.EdmPrimitiveTypeKind;
import org.apache.olingo.commons.api.edm.EdmProperty;
import org.apache.olingo.commons.api.edm.EdmType;
import org.apache.olingo.commons.api.edm.FullQualifiedName;
import org.apache.olingo.server.api.OData;
import org.apache.olingo.server.api.uri.queryoption.AliasQueryOption;
import org.apache.olingo.server.api.uri.queryoption.expression.Expression;
import org.apache.olingo.server.core.uri.parser.UriTokenizer.TokenKind;
import org.apache.olingo.server.core.uri.queryoption.AliasQueryOptionImpl;
import org.apache.olingo.server.core.uri.validator.UriValidationException;
import org.junit.Test;
import org.mockito.Mockito;
public class ExpressionParserTest {
@ -51,6 +67,25 @@ public class ExpressionParserTest {
assertEquals("{null EQ null}", parseExpression("null eq null").toString());
wrongExpression("5 eq '5'");
}
@Test
public void testIntegerTypes() throws Exception {
Expression expression = parseExpression("5 ne 545678979");
assertEquals("{5 NE 545678979}", expression.toString());
expression = parseExpression("5456 eq 5456");
assertEquals("{5456 EQ 5456}", expression.toString());
expression = parseExpression("null ne 54561234567");
assertEquals("{null NE 54561234567}", expression.toString());
expression = parseExpression("null ne 255");
assertEquals("{null NE 255}", expression.toString());
expression = parseExpression("123 le 2551234567890000999999");
assertEquals("{123 LE 2551234567890000999999}", expression.toString());
}
@Test
@ -217,6 +252,12 @@ public class ExpressionParserTest {
wrongExpression("endswith('a',1)");
assertEquals("{concat ['a', 'b']}", parseExpression("concat( 'a' ,\t'b' )").toString());
parseMethod(TokenKind.SubstringofMethod, "'a'", "'b'");
parseMethod(TokenKind.SubstringofMethod, "' '", "'b'");
parseMethod(TokenKind.SubstringofMethod, "' '", "' '");
parseMethod(TokenKind.SubstringofMethod, null, "'a'");
wrongExpression("substringof('a',1)");
}
@Test
@ -239,6 +280,62 @@ public class ExpressionParserTest {
assertEquals("{cast [42, Edm.SByte]}", parseExpression("cast( 42\t, Edm.SByte )").toString());
}
@Test
public void twoParameterAliasMethods() throws Exception {
parseMethodWithParametersWithAlias(TokenKind.SubstringofMethod, "'a'", "'b'");
parseMethodWithParametersWithoutAlias(TokenKind.SubstringofMethod, "'a'", "'b'");
}
private void parseMethodWithParametersWithoutAlias(TokenKind kind, String... parameters)
throws UriParserException, UriValidationException {
final String methodName = kind.name().substring(0, kind.name().indexOf("Method")).toLowerCase(Locale.ROOT)
.replace("geo", "geo.");
String expressionString = methodName + '(';
expressionString += "@word1";
expressionString += ',';
expressionString += parameters[1];
expressionString += ')';
expressionString += "&@word1=" + parameters[0];
Map<String, AliasQueryOption> alias = new HashMap<String, AliasQueryOption>();
AliasQueryOptionImpl aliasQueryOption = new AliasQueryOptionImpl();
aliasQueryOption.setName("@word");
aliasQueryOption.setText("\'a\'");
alias.put("@word", aliasQueryOption);
UriTokenizer tokenizer = new UriTokenizer(expressionString);
final Expression expression = new ExpressionParser(mock(Edm.class), odata).parse(tokenizer, null, null, alias);
assertNotNull(expression);
assertEquals('{' + methodName + ' ' + "[@word1, " + parameters[1] + ']' + '}',
expression.toString());
}
private void parseMethodWithParametersWithAlias(TokenKind kind,
String... parameters) throws UriParserException, UriValidationException {
final String methodName = kind.name().substring(0, kind.name().indexOf("Method")).toLowerCase(Locale.ROOT)
.replace("geo", "geo.");
String expressionString = methodName + '(';
expressionString += "@word";
expressionString += ',';
expressionString += parameters[1];
expressionString += ')';
expressionString += "&@word=" + parameters[0];
Map<String, AliasQueryOption> alias = new HashMap<String, AliasQueryOption>();
AliasQueryOptionImpl aliasQueryOption = new AliasQueryOptionImpl();
aliasQueryOption.setName("@word");
aliasQueryOption.setText("\'a\'");
alias.put("@word", aliasQueryOption);
UriTokenizer tokenizer = new UriTokenizer(expressionString);
final Expression expression = new ExpressionParser(mock(Edm.class), odata).parse(tokenizer, null, null, alias);
assertNotNull(expression);
assertEquals('{' + methodName + ' ' + "[@word, " + parameters[1] + ']' + '}',
expression.toString());
}
private void parseMethod(TokenKind kind, String... parameters) throws UriParserException, UriValidationException {
final String methodName = kind.name().substring(0, kind.name().indexOf("Method")).toLowerCase(Locale.ROOT)
.replace("geo", "geo.");
@ -278,4 +375,309 @@ public class ExpressionParserTest {
assertNotNull(e);
}
}
@Test
public void testPropertyPathExp() throws Exception {
final String entitySetName = "ESName";
final String keyPropertyName = "a";
EdmProperty keyProperty = mockProperty(keyPropertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmKeyPropertyRef keyPropertyRef = mockKeyPropertyRef(keyPropertyName, keyProperty);
EdmEntityType entityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(entityType.getPropertyNames()).thenReturn(Collections.singletonList(keyPropertyName));
Mockito.when(entityType.getProperty(keyPropertyName)).thenReturn(keyProperty);
EdmEntitySet entitySet = mockEntitySet(entitySetName, entityType);
EdmEntityContainer container = mockContainer(entitySetName, entitySet);
Edm mockedEdm = Mockito.mock(Edm.class);
Mockito.when(mockedEdm.getEntityContainer()).thenReturn(container);
UriTokenizer tokenizer = new UriTokenizer("a eq \'abc\'");
final Expression expression = new ExpressionParser(mockedEdm, odata).parse(tokenizer,
entityType, null, null);
assertNotNull(expression);
assertEquals("{[a] EQ \'abc\'}", expression.toString());
}
/**
* @param keyPropertyName
* @param keyPropertyRef
* @return
*/
private EdmEntityType mockEntityType(final String keyPropertyName, EdmKeyPropertyRef keyPropertyRef) {
EdmEntityType entityType = Mockito.mock(EdmEntityType.class);
Mockito.when(entityType.getKeyPredicateNames()).thenReturn(Collections.singletonList(keyPropertyName));
Mockito.when(entityType.getKeyPropertyRefs()).thenReturn(Collections.singletonList(keyPropertyRef));
return entityType;
}
/**
* @param keyPropertyName
* @param keyProperty
* @return
*/
private EdmKeyPropertyRef mockKeyPropertyRef(final String keyPropertyName, EdmProperty keyProperty) {
EdmKeyPropertyRef keyPropertyRef = Mockito.mock(EdmKeyPropertyRef.class);
Mockito.when(keyPropertyRef.getName()).thenReturn(keyPropertyName);
Mockito.when(keyPropertyRef.getProperty()).thenReturn(keyProperty);
return keyPropertyRef;
}
/**
* @param propertyName
* @return
*/
private EdmProperty mockProperty(final String propertyName, final EdmType type) {
EdmProperty keyProperty = Mockito.mock(EdmProperty.class);
Mockito.when(keyProperty.getType()).thenReturn(type);
Mockito.when(keyProperty.getDefaultValue()).thenReturn("");
Mockito.when(keyProperty.getName()).thenReturn(propertyName);
return keyProperty;
}
@Test(expected = UriParserSemanticException.class)
public void testPropertyPathExpWithoutType() throws Exception {
final String entitySetName = "ESName";
final String keyPropertyName = "a";
EdmProperty keyProperty = mockProperty(keyPropertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmKeyPropertyRef keyPropertyRef = mockKeyPropertyRef(keyPropertyName, keyProperty);
EdmEntityType entityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(entityType.getPropertyNames()).thenReturn(Collections.singletonList(keyPropertyName));
Mockito.when(entityType.getProperty(keyPropertyName)).thenReturn(keyProperty);
EdmEntitySet entitySet = mockEntitySet(entitySetName, entityType);
EdmEntityContainer container = mockContainer(entitySetName, entitySet);
Edm mockedEdm = Mockito.mock(Edm.class);
Mockito.when(mockedEdm.getEntityContainer()).thenReturn(container);
UriTokenizer tokenizer = new UriTokenizer("a eq \'abc\'");
new ExpressionParser(mockedEdm, odata).parse(tokenizer, null, null, null);
}
@Test(expected = UriParserSemanticException.class)
public void testPropertyPathExpWithoutProperty() throws Exception {
final String entitySetName = "ESName";
final String keyPropertyName = "a";
EdmProperty keyProperty = mockProperty(keyPropertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmKeyPropertyRef keyPropertyRef = mockKeyPropertyRef(keyPropertyName, keyProperty);
EdmEntityType entityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(entityType.getFullQualifiedName()).thenReturn(new FullQualifiedName("test.ETName"));
EdmEntitySet entitySet = mockEntitySet(entitySetName, entityType);
EdmEntityContainer container = mockContainer(entitySetName, entitySet);
Edm mockedEdm = Mockito.mock(Edm.class);
Mockito.when(mockedEdm.getEntityContainer()).thenReturn(container);
UriTokenizer tokenizer = new UriTokenizer("a eq \'abc\'");
new ExpressionParser(mockedEdm, odata).parse(tokenizer, entityType, null, null);
}
/**
* @param entitySetName
* @param entitySet
* @return
*/
private EdmEntityContainer mockContainer(final String entitySetName, EdmEntitySet entitySet) {
EdmEntityContainer container = Mockito.mock(EdmEntityContainer.class);
Mockito.when(container.getEntitySet(entitySetName)).thenReturn(entitySet);
return container;
}
/**
* @param entitySetName
* @param entityType
* @return
*/
private EdmEntitySet mockEntitySet(final String entitySetName, EdmEntityType entityType) {
EdmEntitySet entitySet = Mockito.mock(EdmEntitySet.class);
Mockito.when(entitySet.getName()).thenReturn(entitySetName);
Mockito.when(entitySet.getEntityType()).thenReturn(entityType);
return entitySet;
}
@Test
public void testComplexPropertyPathExp() throws Exception {
final String entitySetName = "ESName";
final String keyPropertyName = "a";
final String complexPropertyName = "comp";
final String propertyName = "prop";
EdmProperty keyProperty = mockProperty(keyPropertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmKeyPropertyRef keyPropertyRef = mockKeyPropertyRef(keyPropertyName, keyProperty);
EdmProperty property = mockProperty(propertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmComplexType complexType = mockComplexType(propertyName, property);
EdmProperty complexProperty = mockProperty(complexPropertyName, complexType);
EdmEntityType entityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(entityType.getPropertyNames()).thenReturn(Arrays.asList(keyPropertyName, complexPropertyName));
Mockito.when(entityType.getProperty(keyPropertyName)).thenReturn(keyProperty);
Mockito.when(entityType.getProperty(complexPropertyName)).thenReturn(complexProperty);
EdmEntitySet entitySet = mockEntitySet(entitySetName, entityType);
EdmEntityContainer container = mockContainer(entitySetName, entitySet);
Edm mockedEdm = Mockito.mock(Edm.class);
Mockito.when(mockedEdm.getEntityContainer()).thenReturn(container);
UriTokenizer tokenizer = new UriTokenizer("comp/prop eq \'abc\'");
final Expression expression = new ExpressionParser(mockedEdm, odata).parse(tokenizer,
entityType, null, null);
assertNotNull(expression);
assertEquals("{[comp, prop] EQ \'abc\'}", expression.toString());
}
/**
* @param propertyName
* @param property
* @return
*/
private EdmComplexType mockComplexType(final String propertyName, EdmProperty property) {
EdmComplexType complexType = Mockito.mock(EdmComplexType.class);
Mockito.when(complexType.getPropertyNames()).thenReturn(Collections.singletonList(propertyName));
Mockito.when(complexType.getProperty(propertyName)).thenReturn(property);
return complexType;
}
/**
* @param propertyName
* @param property
* @return
*/
private EdmComplexType mockComplexType(final String propertyName, EdmNavigationProperty property) {
EdmComplexType complexType = Mockito.mock(EdmComplexType.class);
Mockito.when(complexType.getPropertyNames()).thenReturn(Collections.singletonList(propertyName));
Mockito.when(complexType.getProperty(propertyName)).thenReturn(property);
return complexType;
}
@Test
public void testLambdaPropertyPathExp() throws Exception {
final String entitySetName = "ESName";
final String keyPropertyName = "a";
final String complexPropertyName = "comp";
final String propertyName = "prop";
EdmProperty keyProperty = mockProperty(keyPropertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmKeyPropertyRef keyPropertyRef = mockKeyPropertyRef(keyPropertyName, keyProperty);
EdmProperty property = mockProperty(propertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmComplexType complexType = mockComplexType(propertyName, property);
EdmProperty complexProperty = mockProperty(complexPropertyName, complexType);
Mockito.when(complexProperty.isCollection()).thenReturn(true);
EdmEntityType entityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(entityType.getPropertyNames()).thenReturn(Arrays.asList(keyPropertyName, complexPropertyName));
Mockito.when(entityType.getProperty(keyPropertyName)).thenReturn(keyProperty);
Mockito.when(entityType.getProperty(complexPropertyName)).thenReturn(complexProperty);
Mockito.when(entityType.getFullQualifiedName()).thenReturn(new FullQualifiedName("test.ET"));
EdmEntitySet entitySet = mockEntitySet(entitySetName, entityType);
EdmEntityContainer container = mockContainer(entitySetName, entitySet);
Edm mockedEdm = Mockito.mock(Edm.class);
Mockito.when(mockedEdm.getEntityContainer()).thenReturn(container);
UriTokenizer tokenizer = new UriTokenizer("comp/any(d:d/prop eq \'abc\')");
Expression expression = new ExpressionParser(mockedEdm, odata).parse(tokenizer,
entityType, null, null);
assertNotNull(expression);
assertEquals("[comp, any]", expression.toString());
tokenizer = new UriTokenizer("comp/all(d:d/prop eq \'abc\')");
expression = new ExpressionParser(mockedEdm, odata).parse(tokenizer,
entityType, null, null);
assertNotNull(expression);
assertEquals("[comp, all]", expression.toString());
}
@Test
public void testNavigationPropertyPathExp() throws Exception {
final String entitySetName = "ESName";
final String keyPropertyName = "a";
final String complexPropertyName = "comp";
final String propertyName = "navProp";
EdmProperty keyProperty = mockProperty(keyPropertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmKeyPropertyRef keyPropertyRef = mockKeyPropertyRef(keyPropertyName, keyProperty);
EdmEntityType targetEntityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(targetEntityType.getFullQualifiedName()).thenReturn(new FullQualifiedName("test.TargetET"));
EdmNavigationProperty navProperty = mockNavigationProperty(propertyName, targetEntityType);
EdmComplexType complexType = mockComplexType(propertyName, navProperty);
EdmProperty complexProperty = mockProperty(complexPropertyName, complexType);
EdmEntityType startEntityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(startEntityType.getFullQualifiedName()).thenReturn(new FullQualifiedName("test.StartET"));
Mockito.when(startEntityType.getPropertyNames()).thenReturn(
Arrays.asList(keyPropertyName, complexPropertyName));
Mockito.when(startEntityType.getProperty(keyPropertyName)).thenReturn(keyProperty);
Mockito.when(startEntityType.getProperty(complexPropertyName)).thenReturn(complexProperty);
EdmEntitySet entitySet = mockEntitySet(entitySetName, startEntityType);
EdmEntityContainer container = mockContainer(entitySetName, entitySet);
Edm mockedEdm = Mockito.mock(Edm.class);
Mockito.when(mockedEdm.getEntityContainer()).thenReturn(container);
UriTokenizer tokenizer = new UriTokenizer("comp/navProp");
final Expression expression = new ExpressionParser(mockedEdm, odata).parse(tokenizer,
startEntityType, null, null);
assertNotNull(expression);
assertEquals("[comp, navProp]", expression.toString());
}
/**
* @param propertyName
* @param entityType
* @return
*/
private EdmNavigationProperty mockNavigationProperty(final String propertyName, EdmEntityType entityType) {
EdmNavigationProperty navProperty = Mockito.mock(EdmNavigationProperty.class);
Mockito.when(navProperty.getName()).thenReturn(propertyName);
Mockito.when(navProperty.getType()).thenReturn(entityType);
return navProperty;
}
@Test
public void testDerivedPathExp() throws Exception {
final String derivedEntitySetName = "ESName";
final String keyPropertyName = "a";
final String propertyName = "navProp";
EdmProperty keyProperty = mockProperty(keyPropertyName,
OData.newInstance().createPrimitiveTypeInstance(EdmPrimitiveTypeKind.String));
EdmKeyPropertyRef keyPropertyRef = mockKeyPropertyRef(keyPropertyName, keyProperty);
EdmEntityType navEntityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(navEntityType.getFullQualifiedName()).thenReturn(new FullQualifiedName("test.navET"));
Mockito.when(navEntityType.getNamespace()).thenReturn("test");
Mockito.when(navEntityType.getPropertyNames()).thenReturn(
Arrays.asList(keyPropertyName));
Mockito.when(navEntityType.getProperty(keyPropertyName)).thenReturn(keyProperty);
EdmEntityType baseEntityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(baseEntityType.getFullQualifiedName()).thenReturn(new FullQualifiedName("test.baseET"));
Mockito.when(baseEntityType.getNamespace()).thenReturn("test");
Mockito.when(baseEntityType.getPropertyNames()).thenReturn(
Arrays.asList(keyPropertyName));
Mockito.when(baseEntityType.getProperty(keyPropertyName)).thenReturn(keyProperty);
Mockito.when(navEntityType.getBaseType()).thenReturn(baseEntityType);
Mockito.when(baseEntityType.compatibleTo(navEntityType)).thenReturn(true);
EdmEntityType entityType = mockEntityType(keyPropertyName, keyPropertyRef);
Mockito.when(entityType.getFullQualifiedName()).thenReturn(new FullQualifiedName("test.derivedET"));
Mockito.when(entityType.getNamespace()).thenReturn("test");
Mockito.when(entityType.getPropertyNames()).thenReturn(Arrays.asList(keyPropertyName, propertyName));
EdmNavigationProperty navProperty = mockNavigationProperty(propertyName, navEntityType);
Mockito.when(entityType.getProperty(propertyName)).thenReturn(navProperty);
EdmEntitySet entitySet = mockEntitySet(derivedEntitySetName, entityType);
EdmEntityContainer container = mockContainer(derivedEntitySetName, entitySet);
Edm mockedEdm = Mockito.mock(Edm.class);
Mockito.when(mockedEdm.getEntityContainer()).thenReturn(container);
Mockito.when(mockedEdm.getEntityType(new FullQualifiedName("test.baseET"))).thenReturn(baseEntityType);
UriTokenizer tokenizer = new UriTokenizer("navProp/test.baseET");
Expression expression = new ExpressionParser(mockedEdm, odata).parse(tokenizer,
entityType, null, null);
assertNotNull(expression);
assertEquals("[navProp]", expression.toString());
}
}

View File

@ -182,6 +182,8 @@ public class ExpressionVisitorImpl implements ExpressionVisitor<VisitorOperand>
return methodCallOperation.floor();
case CEILING:
return methodCallOperation.ceiling();
case SUBSTRINGOF:
return methodCallOperation.substringof();
default:
return throwNotImplemented();

View File

@ -109,6 +109,15 @@ public class MethodCallOperator {
}, primBoolean);
}
public VisitorOperand substringof() throws ODataApplicationException {
return stringFunction(new StringFunction() {
@Override
public Object perform(final List<String> params) {
return params.get(1).contains(params.get(0));
}
}, primBoolean);
}
public VisitorOperand toLower() throws ODataApplicationException {
return stringFunction(new StringFunction() {
@Override

View File

@ -50,6 +50,29 @@ public class ExpressionParserTest {
@Test
public void filter() throws Exception {
testFilter.runOnESCompCollComp("PropertyComp/CollPropertyComp/any"
+ "(f:f/olingo.odata.test1.CTBase/AdditionalPropString eq 'ADD TEST')")
.is("<PropertyComp/CollPropertyComp/<ANY;<<f/olingo.odata.test1.CTBase/AdditionalPropString> eq <'ADD TEST'>>>>")
.isMember().goPath()
.first()
.isComplexProperty("PropertyComp", ComplexTypeProvider.nameCTCompCollComp, false)
.n()
.isComplexProperty("CollPropertyComp", ComplexTypeProvider.nameCTTwoPrim, true)
.n()
.isUriPathInfoKind(UriResourceKind.lambdaAny)
.goLambdaExpression()
.isBinary(BinaryOperatorKind.EQ)
.left().goPath()
.first().isUriPathInfoKind(UriResourceKind.lambdaVariable)
.isType(ComplexTypeProvider.nameCTTwoPrim, false)
.n()
.isPrimitiveProperty("AdditionalPropString", PropertyProvider.nameString, false)
.goUpFilterValidator()
.root().right()
.isLiteral("'ADD TEST'");
testFilter.runOnESCompCollCompNeg("PropertyComp/CollPropertyComp/olingo.odata.test1.BFCCTPrimCompRTESTwoKeyNav()");
testFilter.runOnETAllPrim("PropertyBoolean")
.is("<PropertyBoolean>")
.isType(PropertyProvider.nameBoolean);
@ -1929,6 +1952,24 @@ public class ExpressionParserTest {
.n().isComplexProperty("PropertyComp", ComplexTypeProvider.nameCTAllPrim, false)
.n().isPrimitiveProperty("PropertyDate", PropertyProvider.nameDate, false);
testFilter.runOrderByOnETTwoKeyNav("PropertyComp/PropertyComp/PropertyDate eq "
+ "$root/SINav/PropertyComp/PropertyComp/PropertyDate")
.isSortOrder(0, false)
.goOrder(0).isBinary(BinaryOperatorKind.EQ).left().goPath()
.first().isComplexProperty("PropertyComp", ComplexTypeProvider.nameCTPrimComp, false)
.n().isComplexProperty("PropertyComp", ComplexTypeProvider.nameCTAllPrim, false)
.n().isPrimitiveProperty("PropertyDate", PropertyProvider.nameDate, false)
.goUpFilterValidator()
.goOrder(0).right().goPath()
.first().isUriPathInfoKind(UriResourceKind.root)
.n().isSingleton("SINav")
.n().isComplexProperty("PropertyComp", ComplexTypeProvider.nameCTPrimComp, false)
.n().isComplexProperty("PropertyComp", ComplexTypeProvider.nameCTAllPrim, false)
.n().isPrimitiveProperty("PropertyDate", PropertyProvider.nameDate, false);
testFilter.runOrderByOnETTwoKeyNavNeg("PropertyComp/PropertyComp/PropertyDate eq "
+ "$root/SIType/PropertyComp/PropertyComp/PropertyDate");
testFilter.runOrderByOnETTwoKeyNav("PropertyComp/PropertyComp/PropertyDate eq 2013-11-12 desc,"
+ "PropertyString eq 'SomeString' desc")
.isSortOrder(0, true)

View File

@ -51,6 +51,7 @@ import org.apache.olingo.server.api.uri.queryoption.expression.Unary;
import org.apache.olingo.server.core.uri.UriResourceFunctionImpl;
import org.apache.olingo.server.core.uri.parser.Parser;
import org.apache.olingo.server.core.uri.parser.UriParserException;
import org.apache.olingo.server.core.uri.parser.UriParserSemanticException;
import org.apache.olingo.server.core.uri.queryoption.expression.BinaryImpl;
import org.apache.olingo.server.core.uri.queryoption.expression.MemberImpl;
import org.apache.olingo.server.core.uri.queryoption.expression.MethodImpl;
@ -103,6 +104,16 @@ public class FilterValidator implements TestValidator {
return runUriOrderBy("ESTwoKeyNav", "$orderby=" + orderBy);
}
public FilterValidator runOrderByOnETTwoKeyNavNeg(final String orderBy)
throws UriParserException, UriValidationException {
try {
runUriOrderBy("ESTwoKeyNav", "$orderby=" + orderBy);
} catch (UriParserSemanticException e) {
assertEquals("EntitySet or singleton expected.", e.getMessage());
}
return this;
}
public FilterValidator runOrderByOnETMixEnumDefCollComp(final String orderBy)
throws UriParserException, UriValidationException {
return runUriOrderBy("ESMixEnumDefCollComp", "$orderby=" + orderBy);
@ -134,6 +145,20 @@ public class FilterValidator implements TestValidator {
return runUri("ESAllPrim(1)", "$filter=" + filter);
}
public FilterValidator runOnESCompCollComp(final String filter) throws UriParserException, UriValidationException {
return runUri("ESCompCollComp", "$filter=" + filter);
}
public FilterValidator runOnESCompCollCompNeg(final String filter)
throws UriParserException, UriValidationException {
try {
runUri("ESCompCollComp", "$filter=" + filter);
} catch (UriParserSemanticException e) {
assertEquals("Bound function 'olingo.odata.test1.BFCCTPrimCompRTESTwoKeyNav' not found.", e.getMessage());
}
return this;
}
public FilterValidator runOnETKeyNav(final String filter) throws UriParserException, UriValidationException {
return runUri("ESKeyNav(1)", "$filter=" + filter);
}