blob: ac3eb54b62941413c32c63964d5d73936aea6cbc [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.freemarker.core;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.util.Date;
import org.apache.freemarker.core.model.TemplateBooleanModel;
import org.apache.freemarker.core.model.TemplateDateModel;
import org.apache.freemarker.core.model.TemplateModel;
import org.apache.freemarker.core.model.TemplateNumberModel;
import org.apache.freemarker.core.model.impl.SimpleDate;
import org.apache.freemarker.core.model.impl.SimpleNumber;
import org.apache.freemarker.core.model.impl.SimpleString;
import org.apache.freemarker.core.util._NumberUtils;
import org.apache.freemarker.core.util._StringUtils;
/**
* 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 TemplateException {
final int n;
try {
n = _NumberUtils.toIntExact(num);
} catch (ArithmeticException e) {
throw new TemplateException(target,
"The left side operand value isn't compatible with ?", key, ": ", e.getMessage());
}
if (n <= 0) {
throw new TemplateException(target,
"The left side operand of to ?", key, " must be at least 1, but was ", Integer.valueOf(n), ".");
}
return new SimpleString(toABC(n));
}
protected abstract String toABC(int n);
}
static class lower_abcBI extends abcBI {
@Override
protected String toABC(int n) {
return _StringUtils.toLowerABC(n);
}
}
static class upper_abcBI extends abcBI {
@Override
protected String toABC(int n) {
return _StringUtils.toUpperABC(n);
}
}
static class absBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateException {
if (num instanceof Integer) {
int n = 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 = num.doubleValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof Float) {
float n = num.floatValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof Long) {
long n = num.longValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof Short) {
short n = num.shortValue();
if (n < 0) {
return new SimpleNumber(-n);
} else {
return model;
}
} else if (num instanceof Byte) {
byte n = 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 TemplateException("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 TemplateException {
return _NumberUtils.isInfinite(num) ? TemplateBooleanModel.TRUE : TemplateBooleanModel.FALSE;
}
}
static class is_nanBI extends BuiltInForNumber {
@Override
TemplateModel calculateResult(Number num, TemplateModel model) throws TemplateException {
return _NumberUtils.isNaN(num) ? TemplateBooleanModel.TRUE : TemplateBooleanModel.FALSE;
}
}
// Does both someNumber?long and someDate?long, thus it doesn't extend NumberBuiltIn
static class longBI extends ASTExpBuiltIn {
@Override
TemplateModel _eval(Environment env)
throws TemplateException {
TemplateModel model = target.eval(env);
if (!(model instanceof TemplateNumberModel)
&& model instanceof TemplateDateModel) {
Date date = _EvalUtils.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 TemplateException {
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 long safeToLong(Number num) throws TemplateException {
if (num instanceof Double) {
double d = Math.round(num.doubleValue());
if (d > Long.MAX_VALUE || d < Long.MIN_VALUE) {
throw new TemplateException(
"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(num.floatValue());
if (f > Long.MAX_VALUE || f < Long.MIN_VALUE) {
throw new TemplateException(
"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 TemplateException("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 TemplateException("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 {
// TODO [FM3] Should add Atomic* types in 2.4...
throw new TemplateException("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() { }
}