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

com.alibaba.qlexpress4.runtime.instruction.SpreadMethodInvokeInstruction Maven / Gradle / Ivy

package com.alibaba.qlexpress4.runtime.instruction;

import com.alibaba.qlexpress4.QLOptions;
import com.alibaba.qlexpress4.exception.ErrorReporter;
import com.alibaba.qlexpress4.exception.QLErrorCodes;
import com.alibaba.qlexpress4.runtime.Parameters;
import com.alibaba.qlexpress4.runtime.QContext;
import com.alibaba.qlexpress4.runtime.QResult;
import com.alibaba.qlexpress4.runtime.Value;
import com.alibaba.qlexpress4.runtime.data.DataValue;
import com.alibaba.qlexpress4.runtime.util.MethodInvokeUtils;
import com.alibaba.qlexpress4.utils.PrintlnUtils;

import java.lang.reflect.Array;
import java.util.ArrayList;
import java.util.List;
import java.util.function.Consumer;

/**
 * Operation: Invoke specified method of each object in the list
 * Input: ${argNum} + 1
 * Output: 1, a list composed of return values from methods.
 * Author: DQinYuan
 */
public class SpreadMethodInvokeInstruction extends QLInstruction {

    private final String methodName;

    private final int argNum;

    public SpreadMethodInvokeInstruction(ErrorReporter errorReporter, String methodName, int argNum) {
        super(errorReporter);
        this.methodName = methodName;
        this.argNum = argNum;
    }

    @Override
    public QResult execute(QContext qContext, QLOptions qlOptions) {
        Parameters parameters = qContext.pop(this.argNum + 1);
        Object traversable = parameters.get(0).get();
        if (traversable == null) {
            if (qlOptions.isAvoidNullPointer()) {
                qContext.push(DataValue.NULL_VALUE);
                return QResult.NEXT_INSTRUCTION;
            }
            throw errorReporter.reportFormat(QLErrorCodes.NONTRAVERSABLE_OBJECT.name(),
                    QLErrorCodes.NONTRAVERSABLE_OBJECT.getErrorMsg(),
                    "null");
        }
        Class[] type = new Class[this.argNum];
        Object[] params = new Object[this.argNum];
        for (int i = 0; i < this.argNum; i++) {
            Value v = parameters.get(i + 1);
            params[i] = v.get();
            type[i] = v.getType();
        }

        if (traversable instanceof Iterable) {
            Iterable iterable = (Iterable) traversable;
            List result = new ArrayList<>();
            for (Object item : iterable) {
                if (item == null) {
                    if (qlOptions.isAvoidNullPointer()) {
                        result.add(null);
                        continue;
                    }
                    throw errorReporter.report(new NullPointerException(), QLErrorCodes.NULL_METHOD_ACCESS.name(),
                            QLErrorCodes.NULL_METHOD_ACCESS.getErrorMsg());
                }
                Value invokeRes = MethodInvokeUtils.findMethodAndInvoke(item, methodName, params, type,
                        qContext.getReflectLoader(), errorReporter);
                result.add(invokeRes.get());
            }
            qContext.push(new DataValue(result));
        } else if (traversable.getClass().isArray()) {
            int arrLen = Array.getLength(traversable);
            List result = new ArrayList<>();
            for (int i = 0; i < arrLen; i++) {
                Object item = Array.get(traversable, i);
                if (item == null) {
                    if (qlOptions.isAvoidNullPointer()) {
                        result.add(null);
                        continue;
                    }
                    throw errorReporter.report(new NullPointerException(), QLErrorCodes.NULL_METHOD_ACCESS.name(),
                            QLErrorCodes.NULL_METHOD_ACCESS.getErrorMsg());
                }
                Value invokeRes = MethodInvokeUtils.findMethodAndInvoke(item, methodName, params, type,
                        qContext.getReflectLoader(), errorReporter);
                result.add(invokeRes.get());
            }
            qContext.push(new DataValue(result));
        } else {
            throw errorReporter.reportFormat(QLErrorCodes.NONTRAVERSABLE_OBJECT.name(),
                    QLErrorCodes.NONTRAVERSABLE_OBJECT.getErrorMsg(),
                    traversable.getClass().getName());
        }
        return QResult.NEXT_INSTRUCTION;
    }

    @Override
    public int stackInput() {
        return argNum + 1;
    }

    @Override
    public int stackOutput() {
        return 1;
    }

    @Override
    public void println(int index, int depth, Consumer debug) {
        PrintlnUtils.printlnByCurDepth(depth, index + ": SpreadMethodInvoke " + methodName, debug);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy