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

com.github.basking2.sdsai.itrex.functions.ImportFunction Maven / Gradle / Ivy

package com.github.basking2.sdsai.itrex.functions;

import com.github.basking2.sdsai.itrex.EvaluationContext;
import com.github.basking2.sdsai.itrex.Evaluator;
import com.github.basking2.sdsai.itrex.SExprRuntimeException;
import com.github.basking2.sdsai.itrex.packages.Package;

import java.lang.reflect.Field;
import java.util.Iterator;

/**
 * Import functions into the current runtime.
 *
 * Returns OK on success or an error message.
 */
public class ImportFunction implements FunctionInterface {

    final Evaluator evaluator;
    final EvaluationContext evaluationContext;

    public ImportFunction(
            final Evaluator evaluator,
            final EvaluationContext evaluationContext
    ) {
        this.evaluator = evaluator;
        this.evaluationContext = evaluationContext;
    }

    @Override
    public String apply(final Iterator iterator, final EvaluationContext evaluationContext) {

        final StringBuilder errors = new StringBuilder();

        final Object importMe = iterator.next();

        String destinationPackage = null;

        if (iterator.hasNext()) {
            final String keyWord = iterator.next().toString();
            if (! "as".equals(keyWord)) {
                throw new SExprRuntimeException("Expected \"as\" but got "+keyWord+" in import.");
            }

            if (iterator.hasNext()) {
                destinationPackage = iterator.next().toString();
            }
            else {
                destinationPackage = "";
            }
        }

        final String s = doImport(importMe, destinationPackage);
        if (s != null) {
            errors.append(s).append("\n");
        }

        if (errors.length() > 0) {
            return errors.toString();
        }
        else {
            return "OK";
        }
    }

    /**
     * @param o The object to import.
     *
     * @return Null on success or an error string.
     */
    private String doImport(final Object o, final String destinationPackage) {
        if (o instanceof String) {
            try {
                final Class clazz = getClass().forName((String)o);

                if (Package.class.isAssignableFrom(clazz)) {
                    try {
                        doImport(clazz.newInstance(), destinationPackage);
                    } catch (InstantiationException e) {
                        throw new SExprRuntimeException("Creating "+((String) o), e);
                    } catch (IllegalAccessException e) {
                        throw new SExprRuntimeException("Creating "+((String) o), e);
                    }
                }
                else {
                    doImportStatics(clazz, clazz, destinationPackage);
                }
            }
            catch (final ClassNotFoundException e) {
                return e.getMessage();
            }
        }
        else if (o instanceof Package) {
            final Package p = (Package)o;
            p.importTo(evaluator, destinationPackage);
        }
        else if (o instanceof Class) {
            return doImportStatics((Class)o, o, destinationPackage);
        }
        else {
            return doImportStatics(o.getClass(), o, destinationPackage);
        }

        return null;
    }

    private String doImportStatics(final Class clazz, final Object target, final String destinationPackage) {
        String packageName = null;

        // If the user is not asking we import to a particular package, check for a given package name.
        if (destinationPackage == null) {
            for (final Field field: clazz.getFields()) {
                if (field.getName().equals("__package")) {
                    try {
                        packageName = field.get(target).toString();
                    } catch (IllegalAccessException e) {
                        return e.getMessage();
                    }
                }
            }
        }
        else {
            packageName = destinationPackage;
        }

        // Find a package field.
        for (final Field field : clazz.getFields()) {
            final Object func;
            try {
                func = field.get(target);
            } catch (IllegalAccessException e) {
                return e.getMessage();
            }
            if (func != null && func instanceof FunctionInterface) {
                if (packageName == null || packageName.isEmpty()) {
                    evaluationContext.register(field.getName(), (FunctionInterface) func);
                }
                else {
                    evaluationContext.register(packageName + "." + field.getName(), (FunctionInterface) func);
                }
            }
        }

        return null;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy