Package com.foundationdb.server.types.mcompat.aggr

Source Code of com.foundationdb.server.types.mcompat.aggr.MSum

/**
* Copyright (C) 2009-2013 FoundationDB, LLC
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

package com.foundationdb.server.types.mcompat.aggr;

import com.foundationdb.server.error.OverflowException;
import com.foundationdb.server.types.TAggregator;
import com.foundationdb.server.types.TClass;
import com.foundationdb.server.types.TFixedTypeAggregator;
import com.foundationdb.server.types.TInputSet;
import com.foundationdb.server.types.TInstance;
import com.foundationdb.server.types.TOverloadResult;
import com.foundationdb.server.types.common.BigDecimalWrapper;
import com.foundationdb.server.types.common.types.TBigDecimal;
import com.foundationdb.server.types.mcompat.mtypes.MApproximateNumber;
import com.foundationdb.server.types.mcompat.mtypes.MNumeric;
import com.foundationdb.server.types.texpressions.TInputSetBuilder;
import com.foundationdb.server.types.value.*;
import com.foundationdb.server.types.value.Value;
import com.foundationdb.server.types.value.ValueSource;

import java.util.List;

public class MSum extends TFixedTypeAggregator {

    private final SumType sumType;
   
    private enum SumType {
        BIGINT(MNumeric.BIGINT) {
            @Override
            void input(TInstance type, ValueSource source, TInstance stateType, Value state) {
                long oldState = source.getInt64();
                long input = state.getInt64();
                long sum = oldState + input;
                if (oldState > 0 && input > 0 && sum <= 0) {
                    throw new OverflowException();
                } else if (oldState < 0 && input < 0 && sum >= 0) {
                    throw new OverflowException();
                } else {
                    state.putInt64(sum);
                }
            }
        },
        DOUBLE(MApproximateNumber.DOUBLE) {
            @Override
            void input(TInstance type, ValueSource source, TInstance stateType, Value state) {
                double oldState = source.getDouble();
                double input = state.getDouble();
                double sum = oldState + input;
                if (Double.isInfinite(sum) && !Double.isInfinite(oldState) && !Double.isInfinite(input)) {
                    throw new OverflowException();
                } else {
                    state.putDouble(sum);
                }
            }
        },
        DECIMAL(MNumeric.DECIMAL) {
            @Override
            void input(TInstance type, ValueSource source, TInstance stateType, Value state) {
                BigDecimalWrapper oldState = TBigDecimal.getWrapper(source, type);
                BigDecimalWrapper input = TBigDecimal.getWrapper(state, type);
                state.putObject(oldState.add(input));
            }
        }
        ;
        abstract void input(TInstance type, ValueSource source, TInstance stateType, Value state);
        private final TClass typeClass;
       
        private SumType(TClass typeClass) {
            this.typeClass = typeClass;
        }
    }
   
    public static final TAggregator[] INSTANCES = {
        new MSum(SumType.DECIMAL),
        new MSum(SumType.DOUBLE),
        new MSum(SumType.BIGINT)
    };
   
    private MSum(SumType sumType) {
        super("sum", sumType.typeClass);
        this.sumType = sumType;
    }

    // Want integers to all sum as long and floats as double, but decimals as the
    // particular precision of the input.

    @Override
    public List<TInputSet> inputSets() {
        if (sumType != SumType.DECIMAL)
            return super.inputSets();

        TInputSetBuilder builder = new TInputSetBuilder();
        builder.pickingCovers(inputClass(), 0);
        return builder.toList();
    }
   
    @Override
    public TOverloadResult resultType() {
        if (sumType != SumType.DECIMAL)
            return super.resultType();

        return TOverloadResult.picking();
    }

    @Override
    public void input(TInstance type, ValueSource source, TInstance stateType, Value state, Object o) {
            if (source.isNull())
                return;
        if (!state.hasAnyValue())
            ValueTargets.copyFrom(source, state);
        else
            sumType.input(type, source, stateType, state);
    }

    @Override
    public void emptyValue(ValueTarget state) {
        state.putNull();
    }
}
TOP

Related Classes of com.foundationdb.server.types.mcompat.aggr.MSum

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.