Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix scalar exp encoding #665

Open
wants to merge 10 commits into
base: master
Choose a base branch
from
Original file line number Diff line number Diff line change
Expand Up @@ -983,32 +983,34 @@ final public void stringLiteral() throws ParseException {
case SINGLE_QUOTED_STRING:{
jj_consume_token(SINGLE_QUOTED_STRING);
ExpScalar jjtn001 = new ExpScalar(JJTSCALAR);
boolean jjtc001 = true;
jjtree.openNodeScope(jjtn001);
boolean jjtc001 = true;
jjtree.openNodeScope(jjtn001);
try {
jjtree.closeNodeScope(jjtn001, 0);
jjtc001 = false;
jjtc001 = false;
jjtn001.jjtSetValue(token_source.literalValue);
jjtn001.syncScalarImage("'" + token_source.literalValue + "'");
} finally {
if (jjtc001) {
jjtree.closeNodeScope(jjtn001, 0);
}
jjtree.closeNodeScope(jjtn001, 0);
}
}
break;
}
case DOUBLE_QUOTED_STRING:{
jj_consume_token(DOUBLE_QUOTED_STRING);
ExpScalar jjtn002 = new ExpScalar(JJTSCALAR);
boolean jjtc002 = true;
jjtree.openNodeScope(jjtn002);
boolean jjtc002 = true;
jjtree.openNodeScope(jjtn002);
try {
jjtree.closeNodeScope(jjtn002, 0);
jjtc002 = false;
jjtc002 = false;
jjtn002.jjtSetValue(token_source.literalValue);
jjtn002.syncScalarImage("\"" + token_source.literalValue + "\"");
} finally {
if (jjtc002) {
jjtree.closeNodeScope(jjtn002, 0);
}
jjtree.closeNodeScope(jjtn002, 0);
}
}
break;
}
Expand Down
64 changes: 64 additions & 0 deletions agrest-engine/src/main/java/io/agrest/exp/parser/ExpScalar.java
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,14 @@
/* JavaCCOptions:MULTI=true,NODE_USES_PARSER=false,VISITOR=true,TRACK_TOKENS=false,NODE_PREFIX=Exp,NODE_EXTENDS=,NODE_FACTORY=,SUPPORT_CLASS_VISIBILITY_PUBLIC=true */
package io.agrest.exp.parser;

import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.function.Function;

public class ExpScalar extends ExpBaseScalar<Object> {

protected String scalarImage = String.valueOf(value);

public ExpScalar(int id) {
super(id);
}
Expand All @@ -27,6 +32,39 @@ protected Object transformExpression(Function<Object, Object> transformer) {
return transformer.apply(new ExpScalar(getValue()));
}

protected void syncScalarImage(CharSequence value) {
if (value.length() < 2) {
scalarImage = "'" + value + "'";
return;
}
String stringValue = value.toString();
char firstChar = stringValue.charAt(0);
if (firstChar != '\'' && firstChar != '"') {
scalarImage = "'" + value + "'";
return;
}
String escapedContent = stringValue
.substring(1, stringValue.length() - 1)
.replaceAll(String.valueOf(firstChar), "\\\\" + firstChar);
scalarImage = firstChar + escapedContent + firstChar;
}

protected void syncScalarImage(Long value) {
scalarImage = value + "L";
}

protected void syncScalarImage(BigInteger value) {
scalarImage = value + "H";
}

protected void syncScalarImage(BigDecimal value) {
scalarImage = value + "B";
}

protected void syncScalarImage(Object value) {
scalarImage = String.valueOf(value);
}

/**
* Accept the visitor.
**/
Expand All @@ -35,6 +73,32 @@ public <T> T jjtAccept(AgExpressionParserVisitor<T> visitor, T data) {
return visitor.visit(this, data);
}

@Override
public void jjtSetValue(Object value) {
super.jjtSetValue(value);
if (value == null) {
scalarImage = "null";
return;
}
if (value instanceof CharSequence) {
syncScalarImage(((CharSequence) value));
return;
}
if (value.getClass() == Long.class) {
m-dzianishchyts marked this conversation as resolved.
Show resolved Hide resolved
syncScalarImage(((Long) value));
return;
}
if (value.getClass() == BigInteger.class) {
syncScalarImage(((BigInteger) value));
return;
}
if (value.getClass() == BigDecimal.class) {
syncScalarImage(((BigDecimal) value));
return;
}
syncScalarImage(value);
}

@Override
protected ExpScalar shallowCopy() {
return new ExpScalar(getValue());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,6 @@
import io.agrest.protocol.Exp;

import java.util.Arrays;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;

Expand Down Expand Up @@ -260,9 +258,7 @@ public static String convert(ExpPath exp) {
}

public static String convert(ExpScalar exp) {
return exp.value instanceof CharSequence
? "'" + exp.value + "'"
: String.valueOf(exp.value);
return exp.scalarImage;
}

public static String convert(ExpScalarList exp) {
Expand Down
17 changes: 9 additions & 8 deletions agrest-engine/src/main/java/io/agrest/protocol/Exp.java
Original file line number Diff line number Diff line change
Expand Up @@ -64,26 +64,27 @@ static Exp path(String path) {
*/
static Exp scalar(Object value) {
if (value == null) {
return new ExpScalar(AgExpressionParserTreeConstants.JJTSCALAR);
return new ExpScalar();
}

ExpBaseScalar<?> scalar;
if (value instanceof Collection) {
scalar = new ExpScalarList(AgExpressionParserTreeConstants.JJTSCALARLIST);
} else if (value.getClass().isArray()) {
scalar = new ExpScalarList((Collection<?>) value);
return scalar;
}
if (value.getClass().isArray()) {
Class<?> componentType = value.getClass().getComponentType();
if (componentType.isPrimitive()) {
value = ExpUtils.wrapPrimitiveArray(value);
} else {
value = Arrays.asList((Object[]) value);
}
scalar = new ExpScalarList(AgExpressionParserTreeConstants.JJTSCALARLIST);
} else {
scalar = new ExpScalar(AgExpressionParserTreeConstants.JJTSCALAR);
scalar = new ExpScalarList();
scalar.jjtSetValue(value);
return scalar;
}

scalar.jjtSetValue(value);
return scalar;
return new ExpScalar(value);
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -178,9 +178,17 @@ void stringParameter() : {}

void stringLiteral() : {}
{
<SINGLE_QUOTED_STRING> { jjtThis.jjtSetValue(token_source.literalValue); } #Scalar(0)
<SINGLE_QUOTED_STRING>
{
jjtThis.jjtSetValue(token_source.literalValue);
jjtThis.syncScalarImage("'" + token_source.literalValue + "'");
} #Scalar(0)
|
<DOUBLE_QUOTED_STRING> { jjtThis.jjtSetValue(token_source.literalValue); } #Scalar(0)
<DOUBLE_QUOTED_STRING>
{
jjtThis.jjtSetValue(token_source.literalValue);
jjtThis.syncScalarImage("\"" + token_source.literalValue + "\"");
} #Scalar(0)
}

void stringExpression() : {}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -49,15 +49,13 @@ void parse(String expString) {
"1e1|10.0",
"1E1|10.0",
"1.1e1|11.0",
"3.4028235e+38f|3.4028235E38",
"1.7976931348623157e+308|1.7976931348623157E308",
"1.7976931348623157e+308d|1.7976931348623157E308",
"1_2.1|12.1",
"2.2__34|2.234",
".3e7_6|3.0E75",

// TODO: this is wrong, B suffix expected
"1.7976931348623157e+309b|1.7976931348623157E+309"
"3.4028235e+38f|3.4028235E38",
"1.7976931348623157e+308|1.7976931348623157E308",
"1.7976931348623157e+308d|1.7976931348623157E308",
"1.7976931348623157e+309b|1.7976931348623157E+309B"
})
public void parsedToString(String expString, String expected) {
assertEquals(expected, Exp.parse(expString).toString());
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -50,10 +50,8 @@ void parse(String expString) {
"01_2|10",
"0x1_A|26",
"123___456|123456",

// TODO: this is wrong, L suffix expected in the output
"2147483648L|2147483648",
"9223372036854775808H|9223372036854775808",
"2147483648L|2147483648L",
"9223372036854775808H|9223372036854775808H",
"01234567|342391",
"0x12345678|305419896",
"0x09abcdef|162254319"
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,13 +28,12 @@ void parse(String expString) {
@CsvSource(delimiter = '|', quoteCharacter = 'X', value = {
"'example'|'example'",
" 'example' |'example'",
"\"example\"|'example'",
"\"example\"|\"example\"",
"''|''",
"' '|' '",
"\"\\\"example\\\"\"|'\"example\"'",
// TODO: this is wrong, single quote must be escaped or double quotes used in output
"\"a'b\"|'a'b'",
"'a\"b'|'a\"b'"
"\"\\\"example\\\"\"|\"\\\"example\\\"\"",
"\"a\\\"'b\"|\"a\\\"'b\"",
"'a\"\\'b'|'a\"\\'b'"
})
public void parsedToString(String expString, String expected) {
assertEquals(expected, Exp.parse(expString).toString());
Expand Down
Loading