All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.bigml.binding.timeseries.Forecasts Maven / Gradle / Ivy

package org.bigml.binding.timeseries;

import org.bigml.binding.utils.Utils;
import org.json.simple.JSONArray;
import org.json.simple.JSONObject;

import java.util.ArrayList;
// import java.util.Collections;
// import java.util.List;
// import java.util.Arrays;
// import java.util.Map;
// import java.util.HashMap;
// import java.util.Comparator;

import java.lang.reflect.*;

/**
 * Auxiliary module to store the functions to compute time-series forecasts
 following the formulae in
 https://www.otexts.org/sites/default/files/fpp/images/Table7-8.png
 as explained in https://www.otexts.org/fpp/7/6
**/

public class Forecasts {

    private Double l;
    private Double b;
    private JSONArray s;
    private Double phi;
    private JSONObject submodel;

    private static int[] range(int start, long horizon) {
        int length = (int)horizon;
        int[] range = new int[length];
        for (int i = start; i < length; i++) {
            range[i - start] = i;
        }
        return range;
    }
   
    private static final Double seasonContribution(JSONArray sList, Number step) {
        if (sList.size() > 0) {
            Integer period = sList.size();
            Integer index = Math.abs(1 - period + step.intValue() % period);
            return ((Number)sList.get(index)).doubleValue();
        } else
            return 0.0;
    }

    private static final Float calcPoint(Double op1, Double op2, String seasonality) {
        Double result = 0.0;
        if (seasonality.equals("A")) {
            result = op1 + op2;
        } else if (seasonality.equals("M")) {
            result = op1 * op2;
        } else if (seasonality.equals("N")) {
            result = op1;
        } else {
            assert(false);
        }
        return (Float)((Number)(Math.round(result * 100000) / 100000.0)).floatValue();
    }

    public Forecasts(JSONObject submodel) throws Exception {

        this.submodel = submodel;

        JSONObject finalState = Utils.getFromJSONOr(submodel, "final_state");
        this.l = Utils.getFromJSONOr(finalState, "l", 0.0);
        this.b = Utils.getFromJSONOr(finalState, "b", 0.0);
        this.phi = Utils.getFromJSONOr(finalState, "phi", 0.0);
        this.s = Utils.getFromJSONOr(finalState, "s", new JSONArray());
    }
    
    private final ArrayList trivialForecast(Long horizon, 
                                                    String seasonality) 
        throws Throwable {

        ArrayList points = new ArrayList();
        JSONArray submodelPoints = (JSONArray)Utils.getJSONObject(submodel, "value", null);
        Integer period = submodelPoints != null ? submodelPoints.size() : 0;

/*        if (submodel != null) {
            throw new Exception("TRIVIASL: " + submodel.toString());
        }

        if (submodel == null) {
            throw new Exception("TRIVIAL 1: " + submodel.toString());
        }
*/
        if (period > 1) {
            for (Integer h: range(0, horizon)) {
                points.add((Number)submodelPoints.get(h % period));
            }
        } else {
            for (Integer h: range(0, horizon)) {
                points.add((Number)submodelPoints.get(0));
            }
        }
        return points;
    }

    private final ArrayList naiveForecast(Long horizon, 
                                                  String seasonality) 
    throws Throwable {

        return trivialForecast(horizon, seasonality);
    }

    private final ArrayList meanForecast(Long horizon, 
                                                 String seasonality) 
    throws Throwable {

        return trivialForecast(horizon, seasonality);
    }

    private final ArrayList driftForecast(Long horizon, 
                                                  String seasonality) 
    throws Throwable {

        ArrayList points = new ArrayList();
        for (Integer h: range(0, horizon)) {
            points.add((Number)(Utils.getFromJSONOr(submodel, "value", 0.0).doubleValue() +
                                Utils.getFromJSONOr(submodel, "slope", 0.0).doubleValue() * (h + 1)));
        }
        return points;
    }
   
    private final ArrayList NForecast(Long horizon,
                                              String seasonality)
        throws Throwable {
        
        ArrayList points = new ArrayList();
        for (Integer h: range(0, horizon)) {
            Double sc = seasonContribution(s, h);
            points.add(calcPoint(l, sc, seasonality));
        }
        return points;
    }

    private final ArrayList AForecast(Long horizon,
                                              String seasonality)
        throws Throwable {

        ArrayList points = new ArrayList();
        for (Integer h: range(0, horizon)) {
            Double sc = seasonContribution(s, h);
            Double k = b * (h + 1);
            points.add(calcPoint(l + k, sc, seasonality));
        }
        return points;
    }

    private final ArrayList Ad_Forecast(Long horizon,
                                                String seasonality)
        throws Throwable {

        Double phi_ = phi;
        ArrayList points = new ArrayList();
        for (Integer h: range(0, horizon)) {
            Double sc = seasonContribution(s, h);
            Double k = b * phi_;
            points.add(calcPoint(l * k, sc, seasonality));
            phi_ = phi+ + Math.pow(phi_, h + 2);
        }
        return points;
    }

    private final ArrayList MForecast(Long horizon,
                                              String seasonality)
        throws Throwable {

        ArrayList points = new ArrayList();
        for (Integer h: range(0, horizon)) {
            Double sc = seasonContribution(s, h);
            Double k = Math.pow(b, h + 1);
            points.add(calcPoint(l * k, sc, seasonality));
        }
        return points;
    }

    private final ArrayList MdForecast(Long horizon,
                                               String seasonality)
        throws Throwable {

        Double phi_ = phi;
        ArrayList points = new ArrayList();
        for (Integer h: range(0, horizon)) {
            Double sc = seasonContribution(s, h);
            Double k = Math.pow(b, phi_);
            points.add(calcPoint(l * k, sc, seasonality));
            phi_ = phi+ + Math.pow(phi_, h + 2);
        }
        return points;
    }

    public final ArrayList forecast(String trend,
                                            Long horizon, 
                                            String seasonality) 
        throws Throwable {

        String methodName = trend + "Forecast";
        Method method;
//        try {
            method = this.getClass().getDeclaredMethod(methodName, Long.class, String.class);
            method.setAccessible(true);
//        } 
/*        catch (SecurityException e) {
        } catch (NoSuchMethodException e) {
        }
*/

//        try {
            return (ArrayList)method.invoke(this, horizon, seasonality);
/*        } catch (IllegalArgumentException e) {
        } catch (IllegalAccessException e) {
        } catch (InvocationTargetException e) {
        }
*/        

        /*        
        if (trend.equals("trivial") ||
            trend.equals("naive")  ||
            trend.equals("mean")) {
            return trivialForecast(horizon);
        } else if (trend.equals("drift")) {
            return driftForecast(horizon);
        } else if (trend.equals("N")) {
            return N_Forecast(horizon, seasonality);
        } else if (trend.equals("M")) {
            return M_Forecast(horizon, seasonality);
        } else if (trend.equals("M")) {
            return M_Forecast(horizon, seasonality);
        }

        throw new Exception("Unkwnown Trend: " + trend);
*/    }
 
   
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy