blob: 2a3bb31677c77c608b9a13f94ec4b086e913906d [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 freemarker.core;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Date;
import freemarker.template.SimpleDate;
import freemarker.template.SimpleNumber;
import freemarker.template.SimpleScalar;
import freemarker.template.TemplateBooleanModel;
import freemarker.template.TemplateDateModel;
import freemarker.template.TemplateException;
import freemarker.template.TemplateModel;
import freemarker.template.TemplateModelException;
import freemarker.template.TemplateNumberModel;
import freemarker.template.utility.NumberUtil;
import freemarker.template.utility.StringUtil;
/**
* A holder for builtins that operate exclusively on number left-hand value.
*/
class BuiltInsForNumbers {
private static abstract class abcBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateModelException {
final int n;
try {
n = NumberUtil.toIntExact(num);
} catch (ArithmeticException e) {
throw new _TemplateModelException(target,
"The left side operand value isn't compatible with ?", key, ": ", e.getMessage());
}
if (n <= 0) {
throw new _TemplateModelException(target,
"The left side operand of to ?", key, " must be at least 1, but was ", Integer.valueOf(n), ".");
}
return new SimpleScalar(toABC(n));
}
protected abstract String toABC(int n);
}
static class lower_abcBI extends abcBI {
@Override
protected String toABC(int n) {
return StringUtil.toLowerABC(n);
}
}
static class upper_abcBI extends abcBI {
@Override
protected String toABC(int n) {
return StringUtil.toUpperABC(n);
}
}
static class absBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateModelException {
if (num instanceof Integer) {
int n = ((Integer) num).intValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof BigDecimal) {
BigDecimal n = (BigDecimal) num;
if (n.signum() < 0) {
return new SimpleNumber(n.negate());
} else {
return model;
}
} else if (num instanceof Double) {
double n = ((Double) num).doubleValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof Float) {
float n = ((Float) num).floatValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof Long) {
long n = ((Long) num).longValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof Short) {
short n = ((Short) num).shortValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof Byte) {
byte n = ((Byte) num).byteValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof BigInteger) {
BigInteger n = (BigInteger) num;
if (n.signum() < 0) {
return new SimpleNumber(n.negate());
} else {
return model;
}
} else {
throw new _TemplateModelException("Unsupported number class: ", num.getClass());
}
}
}
static class byteBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) {
if (num instanceof Byte) {
return model;
}
return new SimpleNumber(Byte.valueOf(num.byteValue()));
}
}
static class ceilingBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) {
return new SimpleNumber(new BigDecimal(num.doubleValue()).divide(BIG_DECIMAL_ONE, 0, BigDecimal.ROUND_CEILING));
}
}
static class doubleBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) {
if (num instanceof Double) {
return model;
}
return new SimpleNumber(num.doubleValue());
}
}
static class floatBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) {
if (num instanceof Float) {
return model;
}
return new SimpleNumber(num.floatValue());
}
}
static class floorBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) {
return new SimpleNumber(new BigDecimal(num.doubleValue()).divide(BIG_DECIMAL_ONE, 0, BigDecimal.ROUND_FLOOR));
}
}
static class intBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) {
if (num instanceof Integer) {
return model;
}
return new SimpleNumber(num.intValue());
}
}
static class is_infiniteBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateModelException {
return NumberUtil.isInfinite(num) ? TemplateBooleanModel.TRUE : TemplateBooleanModel.FALSE;
}
}
static class is_nanBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateModelException {
return NumberUtil.isNaN(num) ? TemplateBooleanModel.TRUE : TemplateBooleanModel.FALSE;
}
}
// Does both someNumber?long and someDate?long, thus it doesn't extend NumberBuiltIn
static class longBI extends BuiltIn {
@Override
TemplateModel _eval(Environment env)
throws TemplateException {
TemplateModel model = target.eval(env);
if (!(model instanceof TemplateNumberModel)
&& model instanceof TemplateDateModel) {
Date date = EvalUtil.modelToDate((TemplateDateModel) model, target);
return new SimpleNumber(date.getTime());
} else {
Number num = target.modelToNumber(model, env);
if (num instanceof Long) {
return model;
}
return new SimpleNumber(num.longValue());
}
}
}
static class number_to_dateBI extends BuiltInForNumber {
private final int dateType;
number_to_dateBI(int dateType) {
this.dateType = dateType;
}
@Override
TemplateModel calculateResult(Number num, TemplateModel model)
throws TemplateModelException {
return new SimpleDate(new Date(safeToLong(num)), dateType);
}
}
static class roundBI extends BuiltInForNumber {
private static final BigDecimal half = new BigDecimal("0.5");
@Override
TemplateModel calculateResult(Number num, TemplateModel model) {
return new SimpleNumber(new BigDecimal(num.doubleValue()).add(half).divide(BIG_DECIMAL_ONE, 0, BigDecimal.ROUND_FLOOR));
}
}
static class shortBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) {
if (num instanceof Short) {
return model;
}
return new SimpleNumber(Short.valueOf(num.shortValue()));
}
}
private static final long safeToLong(Number num) throws TemplateModelException {
if (num instanceof Double) {
double d = Math.round(((Double) num).doubleValue());
if (d > Long.MAX_VALUE || d < Long.MIN_VALUE) {
throw new _TemplateModelException(
"Number doesn't fit into a 64 bit signed integer (long): ", Double.valueOf(d));
} else {
return (long) d;
}
} else if (num instanceof Float) {
float f = Math.round(((Float) num).floatValue());
if (f > Long.MAX_VALUE || f < Long.MIN_VALUE) {
throw new _TemplateModelException(
"Number doesn't fit into a 64 bit signed integer (long): ", Float.valueOf(f));
} else {
return (long) f;
}
} else if (num instanceof BigDecimal) {
BigDecimal bd = ((BigDecimal) num).setScale(0, BigDecimal.ROUND_HALF_UP);
if (bd.compareTo(BIG_DECIMAL_LONG_MAX) > 0 || bd.compareTo(BIG_DECIMAL_LONG_MIN) < 0) {
throw new _TemplateModelException("Number doesn't fit into a 64 bit signed integer (long): ", bd);
} else {
return bd.longValue();
}
} else if (num instanceof BigInteger) {
BigInteger bi = (BigInteger) num;
if (bi.compareTo(BIG_INTEGER_LONG_MAX) > 0 || bi.compareTo(BIG_INTEGER_LONG_MIN) < 0) {
throw new _TemplateModelException("Number doesn't fit into a 64 bit signed integer (long): ", bi);
} else {
return bi.longValue();
}
} else if (num instanceof Long || num instanceof Integer || num instanceof Byte || num instanceof Short) {
return num.longValue();
} else {
// Should add Atomic* types in 2.4...
throw new _TemplateModelException("Unsupported number type: ", num.getClass());
}
}
private static final BigDecimal BIG_DECIMAL_ONE = new BigDecimal("1");
private static final BigDecimal BIG_DECIMAL_LONG_MIN = BigDecimal.valueOf(Long.MIN_VALUE);
private static final BigDecimal BIG_DECIMAL_LONG_MAX = BigDecimal.valueOf(Long.MAX_VALUE);
private static final BigInteger BIG_INTEGER_LONG_MIN = BigInteger.valueOf(Long.MIN_VALUE);
private static final BigInteger BIG_INTEGER_LONG_MAX = BigInteger.valueOf(Long.MAX_VALUE);
// Can't be instantiated
private BuiltInsForNumbers() { }
}