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

org.apache.groovy.util.JavaShell Maven / Gradle / Ivy

There is a newer version: 5.0.0-alpha-8
Show newest version
/*
 *  Licensed to the Apache Software Foundation (ASF) under one
 *  or more contributor license agreements.  See the NOTICE file
 *  distributed with this work for additional information
 *  regarding copyright ownership.  The ASF licenses this file
 *  to you 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 org.apache.groovy.util;

import org.apache.groovy.io.StringBuilderWriter;
import org.apache.groovy.lang.annotation.Incubating;
import org.codehaus.groovy.control.CompilerConfiguration;
import org.codehaus.groovy.tools.javac.MemJavaFileObject;

import javax.tools.FileObject;
import javax.tools.ForwardingJavaFileManager;
import javax.tools.JavaCompiler;
import javax.tools.JavaFileManager;
import javax.tools.JavaFileObject;
import javax.tools.SimpleJavaFileObject;
import javax.tools.StandardJavaFileManager;
import javax.tools.ToolProvider;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.lang.reflect.Method;
import java.net.URI;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.charset.Charset;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.Locale;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * A shell for compiling or running pure Java code
 */
@Incubating
public class JavaShell {
    private static final String MAIN_METHOD_NAME = "main";
    private static final URL[] EMPTY_URL_ARRAY = new URL[0];
    private final JavaShellClassLoader jscl;
    private final Locale locale;
    private final Charset charset;

    /**
     * Initializes a newly created {@code JavaShell} object
     */
    public JavaShell() {
        this(null);
    }

    /**
     * Initializes a newly created {@code JavaShell} object
     *
     * @param parentClassLoader the parent class loader for delegation
     */
    public JavaShell(ClassLoader parentClassLoader) {
        jscl = new JavaShellClassLoader(
                EMPTY_URL_ARRAY,
                null == parentClassLoader
                        ? JavaShell.class.getClassLoader()
                        : parentClassLoader
        );

        locale = Locale.ENGLISH;
        charset = Charset.forName(CompilerConfiguration.DEFAULT.getSourceEncoding());
    }

    /**
     * Run main method
     *
     * @param className the main class name
     * @param options compiler options
     * @param src the source code
     * @param args arguments for main method
     */
    public void run(String className, Iterable options, String src, String... args) throws Throwable {
        Class c = compile(className, options, src);
        Method mainMethod = c.getMethod(MAIN_METHOD_NAME, String[].class);
        mainMethod.invoke(null, (Object) args);
    }

    /**
     * Run main method
     *
     * @param className the main class name
     * @param src the source code
     * @param args arguments for main method
     */
    public void run(String className, String src, String... args) throws Throwable {
        run(className, Collections.emptyList(), src, args);
    }

    /**
     * Compile and return the main class
     * @param className the main class name
     * @param options compiler options
     * @param src the source code
     * @return the main class
     */
    public Class compile(final String className, Iterable options, String src) throws IOException, ClassNotFoundException {
        doCompile(className, src, options);

        return jscl.findClass(className);
    }

    /**
     * Compile and return the main class
     * @param className the main class name
     * @param src the source code
     * @return the main class
     */
    public Class compile(final String className, String src) throws IOException, ClassNotFoundException {
        return compile(className, Collections.emptyList(), src);
    }

    /**
     * Compile and return all classes
     *
     * @param className the main class name
     * @param options compiler options
     * @param src the source code
     * @return all classes
     */
    public Map> compileAll(final String className, Iterable options, String src) throws IOException, ClassNotFoundException {
        doCompile(className, src, options);

        Map> classes = new LinkedHashMap<>();
        for (String cn : jscl.getClassMap().keySet()) {
            Class c = jscl.findClass(cn);
            classes.put(cn, c);
        }

        return classes;
    }

    /**
     * Compile and return all classes
     *
     * @param className the main class name
     * @param src the source code
     * @return all classes
     */
    public Map> compileAll(final String className, String src) throws IOException, ClassNotFoundException {
        return compileAll(className, Collections.emptyList(), src);
    }

    private void doCompile(String className, String src, Iterable options) throws IOException {
        JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
        try (BytesJavaFileManager bjfm = new BytesJavaFileManager(compiler.getStandardFileManager(null, locale, charset))) {
            StringBuilderWriter out = new StringBuilderWriter();
            JavaCompiler.CompilationTask task =
                    compiler.getTask(
                            out,
                            bjfm,
                            null,
                            options,
                            Collections.emptyList(),
                            Collections.singletonList(
                                    new MemJavaFileObject(className, src)
                            )
                    );

            task.call();

            if (bjfm.isEmpty()) {
                throw new JavaShellCompilationException(out.toString());
            }

            final Map classMap = bjfm.getClassMap();

            jscl.setClassMap(classMap);
        }
    }

    /**
     * When and only when {@link #compile(String, String)} or {@link #compileAll(String, String)} is invoked,
     * returned class loader will reference the compiled classes.
     */
    public JavaShellClassLoader getClassLoader() {
        return jscl;
    }

    private static final class JavaShellClassLoader extends URLClassLoader {
        private Map classMap = Collections.emptyMap();
        private final Map> classCache = new ConcurrentHashMap<>();

        public JavaShellClassLoader(URL[] urls, ClassLoader parent) {
            super(urls, parent);
        }

        @Override
        public Class findClass(String name) throws ClassNotFoundException {
            final byte[] bytes = classMap.get(name);

            if (null != bytes) {
                return classCache.computeIfAbsent(name, n -> defineClass(n, bytes, 0, bytes.length));
            }

            return super.findClass(name);
        }

        public Map getClassMap() {
            return classMap;
        }

        public void setClassMap(Map classMap) {
            this.classMap = classMap;
        }
    }

    private static final class BytesJavaFileManager extends ForwardingJavaFileManager {
        private final Map fileObjectMap = new HashMap<>();
        private Map classMap;

        public BytesJavaFileManager(StandardJavaFileManager sjfm) {
            super(sjfm);
        }

        public boolean isEmpty() {
            return fileObjectMap.isEmpty();
        }

        @Override
        public JavaFileObject getJavaFileForOutput(
                JavaFileManager.Location location,
                String className,
                JavaFileObject.Kind kind,
                FileObject sibling) {
            BytesJavaFileObject bjfo = new BytesJavaFileObject(className, kind);
            fileObjectMap.put(className, bjfo);
            return bjfo;
        }

        public Map getClassMap() {
            if (classMap != null) return classMap;

            classMap = new LinkedHashMap<>();
            fileObjectMap.forEach((key, value) -> classMap.put(key, value.getBytes()));

            return Collections.unmodifiableMap(classMap);
        }
    }

    private static class BytesJavaFileObject extends SimpleJavaFileObject {
        private final ByteArrayOutputStream baos = new ByteArrayOutputStream();

        public BytesJavaFileObject(String name, Kind kind) {
            super(URI.create("string:///" + name.replace('.', '/') + kind.extension), kind);
        }

        @Override
        public OutputStream openOutputStream() {
            return baos;
        }

        public byte[] getBytes() {
            return baos.toByteArray();
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy