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

com.healthmarketscience.jackcess.impl.expr.FunctionSupport Maven / Gradle / Ivy

There is a newer version: 4.0.7
Show newest version
/*
Copyright (c) 2018 James Ahlborn

Licensed 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 com.healthmarketscience.jackcess.impl.expr;

import java.util.Arrays;

import com.healthmarketscience.jackcess.expr.EvalContext;
import com.healthmarketscience.jackcess.expr.EvalException;
import com.healthmarketscience.jackcess.expr.Function;
import com.healthmarketscience.jackcess.expr.LocaleContext;
import com.healthmarketscience.jackcess.expr.Value;

/**
 *
 * @author James Ahlborn
 */
public class FunctionSupport
{
  private static final char NON_VAR_SUFFIX = '$';

  private FunctionSupport() {}

  public static abstract class BaseFunction implements Function
  {
    private final String _name;
    private final int _minParams;
    private final int _maxParams;

    protected BaseFunction(String name, int minParams, int maxParams)
    {
      _name = name;
      _minParams = minParams;
      _maxParams = maxParams;
    }

    @Override
    public String getName() {
      return _name;
    }

    @Override
    public boolean isPure() {
      // most functions are probably pure, so make this the default
      return true;
    }

    protected void validateNumParams(Value[] params) {
      int num = params.length;
      if((num < _minParams) || (num > _maxParams)) {
        String range = ((_minParams == _maxParams) ? "" + _minParams :
                        _minParams + " to " + _maxParams);
        throw new EvalException(
            "Invalid number of parameters " +
            num + " passed, expected " + range);
      }
    }

    protected EvalException invalidFunctionCall(
        Throwable t, Value[] params)
    {
      String paramStr = Arrays.toString(params);
      String msg = "Invalid function call {" + _name + "(" +
        paramStr.substring(1, paramStr.length() - 1) + ")}";
      return new EvalException(msg, t);
    }

    @Override
    public String toString() {
      return getName() + "()";
    }
  }

  public static abstract class Func0 extends BaseFunction
  {
    protected Func0(String name) {
      super(name, 0, 0);
    }

    @Override
    public boolean isPure() {
      // 0-arg functions are usually not pure
      return false;
    }

    @Override
    public final Value eval(EvalContext ctx, Value... params) {
      try {
        validateNumParams(params);
        return eval0(ctx);
      } catch(Exception e) {
        throw invalidFunctionCall(e, params);
      }
    }

    protected abstract Value eval0(EvalContext ctx);
  }

  public static abstract class Func1 extends BaseFunction
  {
    protected Func1(String name) {
      super(name, 1, 1);
    }

    @Override
    public final Value eval(EvalContext ctx, Value... params) {
      try {
        validateNumParams(params);
        return eval1(ctx, params[0]);
      } catch(Exception e) {
        throw invalidFunctionCall(e, params);
      }
    }

    protected abstract Value eval1(EvalContext ctx, Value param);
  }

  public static abstract class Func1NullIsNull extends BaseFunction
  {
    protected Func1NullIsNull(String name) {
      super(name, 1, 1);
    }

    @Override
    public final Value eval(EvalContext ctx, Value... params) {
      try {
        validateNumParams(params);
        Value param1 = params[0];
        if(param1.isNull()) {
          return param1;
        }
        return eval1(ctx, param1);
      } catch(Exception e) {
        throw invalidFunctionCall(e, params);
      }
    }

    protected abstract Value eval1(EvalContext ctx, Value param);
  }

  public static abstract class Func2 extends BaseFunction
  {
    protected Func2(String name) {
      super(name, 2, 2);
    }

    @Override
    public final Value eval(EvalContext ctx, Value... params) {
      try {
        validateNumParams(params);
        return eval2(ctx, params[0], params[1]);
      } catch(Exception e) {
        throw invalidFunctionCall(e, params);
      }
    }

    protected abstract Value eval2(EvalContext ctx, Value param1, Value param2);
  }

  public static abstract class Func3 extends BaseFunction
  {
    protected Func3(String name) {
      super(name, 3, 3);
    }

    @Override
    public final Value eval(EvalContext ctx, Value... params) {
      try {
        validateNumParams(params);
        return eval3(ctx, params[0], params[1], params[2]);
      } catch(Exception e) {
        throw invalidFunctionCall(e, params);
      }
    }

    protected abstract Value eval3(EvalContext ctx,
                                   Value param1, Value param2, Value param3);
  }

  public static abstract class FuncVar extends BaseFunction
  {
    protected FuncVar(String name) {
      super(name, 0, Integer.MAX_VALUE);
    }

    protected FuncVar(String name, int minParams, int maxParams) {
      super(name, minParams, maxParams);
    }

    @Override
    public final Value eval(EvalContext ctx, Value... params) {
      try {
        validateNumParams(params);
        return evalVar(ctx, params);
      } catch(Exception e) {
        throw invalidFunctionCall(e, params);
      }
    }

    protected abstract Value evalVar(EvalContext ctx, Value[] params);
  }

  public static class StringFuncWrapper implements Function
  {
    private final String _name;
    private final Function _delegate;

    public StringFuncWrapper(Function delegate) {
      _delegate = delegate;
      _name = _delegate.getName() + NON_VAR_SUFFIX;
    }

    @Override
    public String getName() {
      return _name;
    }

    @Override
    public boolean isPure() {
      return _delegate.isPure();
    }

    @Override
    public Value eval(EvalContext ctx, Value... params) {
      Value result = _delegate.eval(ctx, params);
      if(result.isNull()) {
        // non-variant version does not do null-propagation, so force
        // exception to be thrown here
        result.getAsString(ctx);
      }
      return result;
    }

    @Override
    public String toString() {
      return getName() + "()";
    }
  }

  public static boolean getOptionalBooleanParam(
      LocaleContext ctx, Value[] params, int idx) {
    if(params.length > idx) {
      return params[idx].getAsBoolean(ctx);
    }
    return false;
  }

  public static double getOptionalDoubleParam(
      EvalContext ctx, Value[] params, int idx, double defValue) {
    if(params.length > idx) {
      return params[idx].getAsDouble(ctx);
    }
    return defValue;
  }

  public static int getOptionalIntParam(
      LocaleContext ctx, Value[] params, int idx, int defValue) {
    return getOptionalIntParam(ctx, params, idx, defValue, defValue);
  }

  public static int getOptionalIntParam(
      LocaleContext ctx, Value[] params, int idx, int defValue, int useDefValue) {
    int val = defValue;
    if(params.length > idx) {
      val = params[idx].getAsLongInt(ctx);
      if(val == useDefValue) {
        val = defValue;
      }
    }
    return val;
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy