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

fish.payara.micro.boot.loader.Launcher Maven / Gradle / Ivy

/*
 * Copyright 2012-2016 the original author or authors.
 *
 * 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.
 */
// Modified 2017 Payara Foundation
package fish.payara.micro.boot.loader;

import java.io.File;
import java.net.URI;
import java.net.URL;
import java.security.CodeSource;
import java.security.ProtectionDomain;
import java.util.ArrayList;
import java.util.List;

import fish.payara.micro.boot.loader.archive.Archive;
import fish.payara.micro.boot.loader.archive.JarFileArchive;

/**
 * Base class for launchers that can start an application with a fully
 * configured classpath backed by one or more {@link Archive}s.
 *
 * @author Phillip Webb
 * @author Dave Syer
 */
public abstract class Launcher {

    /**
     * Launch the application. This method is the initial entry point that
     * should be called by a subclass
     * {@code public static void main(String[] args)} method.
     *
     * @param args the incoming arguments
     * @throws Exception if the application fails to launch
     */
    protected Object launch(String method, String[] args) throws Exception {
        boolean explode = true;
        String unpackDir = null;
        for (int i = 0; i < args.length; i++) {
            if ("--nested".equals(args[i].toLowerCase())) {
                explode = false;
            } else if ("--unpackdir".equals(args[i].toLowerCase()) || "--rootdir".equals(args[i].toLowerCase())) {
                if (args.length >= (i + 1)) {
                    unpackDir = args[i + 1];
                }
            }
        }

        ClassLoader classLoader;
        if (!explode) {
            classLoader = createClassLoader(getClassPathArchives());
            fish.payara.micro.boot.loader.jar.JarFile.registerUrlProtocolHandler();
        } else {
            if (unpackDir != null) {
                classLoader = new ExplodedURLClassloader(new File(unpackDir));
            } else {
                ExplodedURLClassloader tempDirClassloader = new ExplodedURLClassloader();
                classLoader = tempDirClassloader;
                tempDirClassloader.registerDeleteOnExit();
            }
        }
        return launch(method, args, getMainClass(), classLoader);
    }

    /**
     * Create a classloader for the specified archives.
     *
     * @param archives the archives
     * @return the classloader
     * @throws Exception if the classloader cannot be created
     */
    protected ClassLoader createClassLoader(List archives) throws Exception {
        List urls = new ArrayList(archives.size());
        for (Archive archive : archives) {
            urls.add(archive.getUrl());
        }
        return createClassLoader(urls.toArray(new URL[urls.size()]));
    }

    /**
     * Create a classloader for the specified URLs.
     *
     * @param urls the URLs
     * @return the classloader
     * @throws Exception if the classloader cannot be created
     */
    protected ClassLoader createClassLoader(URL[] urls) throws Exception {
        return new LaunchedURLClassLoader(urls, getClass().getClassLoader());
    }

    /**
     * Launch the application given the archive file and a fully configured
     * classloader.
     *
     * @param args the incoming arguments
     * @param mainClass the main class to run
     * @param classLoader the classloader
     * @throws Exception if the launch fails
     */
    protected Object launch(String method, String[] args, String mainClass, ClassLoader classLoader)
            throws Exception {
        Thread.currentThread().setContextClassLoader(classLoader);
        return createMainMethodRunner(mainClass, method, args, classLoader).run();
    }

    /**
     * Create the {@code MainMethodRunner} used to launch the application.
     *
     * @param mainClass the main class
     * @param args the incoming arguments
     * @param classLoader the classloader
     * @return the main method runner
     */
    protected MainMethodRunner createMainMethodRunner(String mainClass, String method, String[] args,
            ClassLoader classLoader) {
        return new MainMethodRunner(mainClass, args, method);
    }

    /**
     * Returns the main class that should be launched.
     *
     * @return the name of the main class
     * @throws Exception if the main class cannot be obtained
     */
    protected abstract String getMainClass() throws Exception;

    /**
     * Returns the archives that will be used to construct the class path.
     *
     * @return the class path archives
     * @throws Exception if the class path archives cannot be obtained
     */
    protected abstract List getClassPathArchives() throws Exception;

    protected final Archive createArchive() throws Exception {
        ProtectionDomain protectionDomain = getClass().getProtectionDomain();
        CodeSource codeSource = protectionDomain.getCodeSource();
        URI location = (codeSource == null ? null : codeSource.getLocation().toURI());
        String path = (location == null ? null : location.getSchemeSpecificPart());
        if (path == null) {
            throw new IllegalStateException("Unable to determine code source archive");
        }
        File root = new File(path);
        if (!root.exists()) {
            throw new IllegalStateException(
                    "Unable to determine code source archive from " + root);
        }
        return new JarFileArchive(root);
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy