mirror of https://github.com/apache/druid.git
Refactor boolean cast code, add tests (#3016)
This commit is contained in:
parent
70e83bea6d
commit
87c61fa749
|
@ -19,6 +19,7 @@
|
||||||
|
|
||||||
package io.druid.math.expr;
|
package io.druid.math.expr;
|
||||||
|
|
||||||
|
import com.google.common.base.Strings;
|
||||||
import io.druid.common.guava.GuavaUtils;
|
import io.druid.common.guava.GuavaUtils;
|
||||||
import io.druid.java.util.common.logger.Logger;
|
import io.druid.java.util.common.logger.Logger;
|
||||||
|
|
||||||
|
@ -81,4 +82,34 @@ public class Evals
|
||||||
return binary; // best effort.. keep it working
|
return binary; // best effort.. keep it working
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static long asLong(boolean x)
|
||||||
|
{
|
||||||
|
return x ? 1L : 0L;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static double asDouble(boolean x)
|
||||||
|
{
|
||||||
|
return x ? 1D : 0D;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String asString(boolean x)
|
||||||
|
{
|
||||||
|
return String.valueOf(x);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean asBoolean(long x)
|
||||||
|
{
|
||||||
|
return x > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean asBoolean(double x)
|
||||||
|
{
|
||||||
|
return x > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean asBoolean(String x)
|
||||||
|
{
|
||||||
|
return !Strings.isNullOrEmpty(x) && Boolean.valueOf(x);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,7 +23,6 @@ import com.google.common.math.LongMath;
|
||||||
import io.druid.java.util.common.IAE;
|
import io.druid.java.util.common.IAE;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Objects;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*/
|
*/
|
||||||
|
@ -242,13 +241,9 @@ class UnaryNotExpr extends UnaryExpr
|
||||||
public ExprEval eval(ObjectBinding bindings)
|
public ExprEval eval(ObjectBinding bindings)
|
||||||
{
|
{
|
||||||
ExprEval ret = expr.eval(bindings);
|
ExprEval ret = expr.eval(bindings);
|
||||||
if (ret.type() == ExprType.LONG) {
|
// conforming to other boolean-returning binary operators
|
||||||
return ExprEval.of(ret.asBoolean() ? 0L : 1L);
|
ExprType retType = ret.type() == ExprType.DOUBLE ? ExprType.DOUBLE : ExprType.LONG;
|
||||||
}
|
return ExprEval.of(!ret.asBoolean(), retType);
|
||||||
if (ret.type() == ExprType.DOUBLE) {
|
|
||||||
return ExprEval.of(ret.asBoolean() ? 0.0d :1.0d);
|
|
||||||
}
|
|
||||||
throw new IllegalArgumentException("unsupported type " + ret.type());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -458,19 +453,19 @@ class BinLtExpr extends BinaryEvalOpExprBase
|
||||||
@Override
|
@Override
|
||||||
protected ExprEval evalString(String left, String right)
|
protected ExprEval evalString(String left, String right)
|
||||||
{
|
{
|
||||||
return ExprEval.of(left.compareTo(right) < 0 ? 1L : 0L);
|
return ExprEval.of(left.compareTo(right) < 0, ExprType.LONG);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final long evalLong(long left, long right)
|
protected final long evalLong(long left, long right)
|
||||||
{
|
{
|
||||||
return left < right ? 1L : 0L;
|
return Evals.asLong(left < right);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final double evalDouble(double left, double right)
|
protected final double evalDouble(double left, double right)
|
||||||
{
|
{
|
||||||
return left < right ? 1.0d : 0.0d;
|
return Evals.asDouble(left < right);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -484,19 +479,19 @@ class BinLeqExpr extends BinaryEvalOpExprBase
|
||||||
@Override
|
@Override
|
||||||
protected ExprEval evalString(String left, String right)
|
protected ExprEval evalString(String left, String right)
|
||||||
{
|
{
|
||||||
return ExprEval.of(left.compareTo(right) <= 0 ? 1L : 0L);
|
return ExprEval.of(left.compareTo(right) <= 0, ExprType.LONG);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final long evalLong(long left, long right)
|
protected final long evalLong(long left, long right)
|
||||||
{
|
{
|
||||||
return left <= right ? 1L : 0L;
|
return Evals.asLong(left <= right);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final double evalDouble(double left, double right)
|
protected final double evalDouble(double left, double right)
|
||||||
{
|
{
|
||||||
return left <= right ? 1.0d : 0.0d;
|
return Evals.asDouble(left <= right);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -510,19 +505,19 @@ class BinGtExpr extends BinaryEvalOpExprBase
|
||||||
@Override
|
@Override
|
||||||
protected ExprEval evalString(String left, String right)
|
protected ExprEval evalString(String left, String right)
|
||||||
{
|
{
|
||||||
return ExprEval.of(left.compareTo(right) > 0 ? 1L : 0L);
|
return ExprEval.of(left.compareTo(right) > 0, ExprType.LONG);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final long evalLong(long left, long right)
|
protected final long evalLong(long left, long right)
|
||||||
{
|
{
|
||||||
return left > right ? 1L : 0L;
|
return Evals.asLong(left > right);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final double evalDouble(double left, double right)
|
protected final double evalDouble(double left, double right)
|
||||||
{
|
{
|
||||||
return left > right ? 1.0d : 0.0d;
|
return Evals.asDouble(left > right);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -536,19 +531,19 @@ class BinGeqExpr extends BinaryEvalOpExprBase
|
||||||
@Override
|
@Override
|
||||||
protected ExprEval evalString(String left, String right)
|
protected ExprEval evalString(String left, String right)
|
||||||
{
|
{
|
||||||
return ExprEval.of(left.compareTo(right) >= 0 ? 1L : 0L);
|
return ExprEval.of(left.compareTo(right) >= 0, ExprType.LONG);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final long evalLong(long left, long right)
|
protected final long evalLong(long left, long right)
|
||||||
{
|
{
|
||||||
return left >= right ? 1L : 0L;
|
return Evals.asLong(left >= right);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final double evalDouble(double left, double right)
|
protected final double evalDouble(double left, double right)
|
||||||
{
|
{
|
||||||
return left >= right ? 1.0d : 0.0d;
|
return Evals.asDouble(left >= right);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -562,19 +557,19 @@ class BinEqExpr extends BinaryEvalOpExprBase
|
||||||
@Override
|
@Override
|
||||||
protected ExprEval evalString(String left, String right)
|
protected ExprEval evalString(String left, String right)
|
||||||
{
|
{
|
||||||
return ExprEval.of(left.equals(right) ? 1L : 0L);
|
return ExprEval.of(left.equals(right), ExprType.LONG);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final long evalLong(long left, long right)
|
protected final long evalLong(long left, long right)
|
||||||
{
|
{
|
||||||
return left == right ? 1L : 0L;
|
return Evals.asLong(left == right);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final double evalDouble(double left, double right)
|
protected final double evalDouble(double left, double right)
|
||||||
{
|
{
|
||||||
return left == right ? 1.0d : 0.0d;
|
return Evals.asDouble(left == right);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -588,19 +583,19 @@ class BinNeqExpr extends BinaryEvalOpExprBase
|
||||||
@Override
|
@Override
|
||||||
protected ExprEval evalString(String left, String right)
|
protected ExprEval evalString(String left, String right)
|
||||||
{
|
{
|
||||||
return ExprEval.of(!Objects.equals(left, right) ? 1L : 0L);
|
return ExprEval.of(!left.equals(right), ExprType.LONG);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final long evalLong(long left, long right)
|
protected final long evalLong(long left, long right)
|
||||||
{
|
{
|
||||||
return left != right ? 1L : 0L;
|
return Evals.asLong(left != right);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected final double evalDouble(double left, double right)
|
protected final double evalDouble(double left, double right)
|
||||||
{
|
{
|
||||||
return left != right ? 1.0d : 0.0d;
|
return Evals.asDouble(left != right);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -55,9 +55,9 @@ public abstract class ExprEval<T>
|
||||||
{
|
{
|
||||||
switch (type) {
|
switch (type) {
|
||||||
case DOUBLE:
|
case DOUBLE:
|
||||||
return ExprEval.of(value ? 1D : 0D);
|
return ExprEval.of(Evals.asDouble(value));
|
||||||
case LONG:
|
case LONG:
|
||||||
return ExprEval.of(value ? 1L : 0L);
|
return ExprEval.of(Evals.asLong(value));
|
||||||
case STRING:
|
case STRING:
|
||||||
return ExprEval.of(String.valueOf(value));
|
return ExprEval.of(String.valueOf(value));
|
||||||
default:
|
default:
|
||||||
|
@ -162,7 +162,7 @@ public abstract class ExprEval<T>
|
||||||
@Override
|
@Override
|
||||||
public final boolean asBoolean()
|
public final boolean asBoolean()
|
||||||
{
|
{
|
||||||
return asDouble() > 0;
|
return Evals.asBoolean(asDouble());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -202,7 +202,7 @@ public abstract class ExprEval<T>
|
||||||
@Override
|
@Override
|
||||||
public final boolean asBoolean()
|
public final boolean asBoolean()
|
||||||
{
|
{
|
||||||
return asLong() > 0;
|
return Evals.asBoolean(asLong());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -266,7 +266,7 @@ public abstract class ExprEval<T>
|
||||||
@Override
|
@Override
|
||||||
public final boolean asBoolean()
|
public final boolean asBoolean()
|
||||||
{
|
{
|
||||||
return Boolean.valueOf(value);
|
return Evals.asBoolean(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -143,4 +143,35 @@ public class EvalTest
|
||||||
Assert.assertEquals("NULL", eval("nvl(if(x == 9223372036854775807, '', 'x'), 'NULL')", bindings).asString());
|
Assert.assertEquals("NULL", eval("nvl(if(x == 9223372036854775807, '', 'x'), 'NULL')", bindings).asString());
|
||||||
Assert.assertEquals("x", eval("nvl(if(x == 9223372036854775806, '', 'x'), 'NULL')", bindings).asString());
|
Assert.assertEquals("x", eval("nvl(if(x == 9223372036854775806, '', 'x'), 'NULL')", bindings).asString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBooleanReturn()
|
||||||
|
{
|
||||||
|
Expr.ObjectBinding bindings = Parser.withMap(
|
||||||
|
ImmutableMap.of("x", 100L, "y", 100L, "z", 100D, "w", 100D)
|
||||||
|
);
|
||||||
|
ExprEval eval = Parser.parse("x==y").eval(bindings);
|
||||||
|
Assert.assertTrue(eval.asBoolean());
|
||||||
|
Assert.assertEquals(ExprType.LONG, eval.type());
|
||||||
|
|
||||||
|
eval = Parser.parse("x!=y").eval(bindings);
|
||||||
|
Assert.assertFalse(eval.asBoolean());
|
||||||
|
Assert.assertEquals(ExprType.LONG, eval.type());
|
||||||
|
|
||||||
|
eval = Parser.parse("x==z").eval(bindings);
|
||||||
|
Assert.assertTrue(eval.asBoolean());
|
||||||
|
Assert.assertEquals(ExprType.DOUBLE, eval.type());
|
||||||
|
|
||||||
|
eval = Parser.parse("x!=z").eval(bindings);
|
||||||
|
Assert.assertFalse(eval.asBoolean());
|
||||||
|
Assert.assertEquals(ExprType.DOUBLE, eval.type());
|
||||||
|
|
||||||
|
eval = Parser.parse("z==w").eval(bindings);
|
||||||
|
Assert.assertTrue(eval.asBoolean());
|
||||||
|
Assert.assertEquals(ExprType.DOUBLE, eval.type());
|
||||||
|
|
||||||
|
eval = Parser.parse("z!=w").eval(bindings);
|
||||||
|
Assert.assertFalse(eval.asBoolean());
|
||||||
|
Assert.assertEquals(ExprType.DOUBLE, eval.type());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue