edu.umass.cs.gnsserver.installer.EC2Runner.java Source code

Java tutorial

Introduction

Here is the source code for edu.umass.cs.gnsserver.installer.EC2Runner.java

Source

/*
 *
 *  Copyright (c) 2015 University of Massachusetts
 *
 *  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.
 *
 *  Initial developer(s): Westy
 *
 */
package edu.umass.cs.gnsserver.installer;

import com.amazonaws.auth.AWSCredentials;
import com.amazonaws.auth.PropertiesCredentials;
import com.amazonaws.services.ec2.AmazonEC2;
import com.amazonaws.services.ec2.AmazonEC2Client;
import com.amazonaws.services.ec2.model.Instance;
import com.amazonaws.services.ec2.model.Tag;

import edu.umass.cs.aws.support.AMIRecord;
import edu.umass.cs.aws.support.AMIRecordType;
import edu.umass.cs.aws.support.AWSEC2;
import edu.umass.cs.aws.support.InstanceStateRecord;
import edu.umass.cs.aws.support.RegionRecord;
import edu.umass.cs.gnsserver.main.GNSConfig;
import edu.umass.cs.gnsserver.utils.GEOLocator;
import edu.umass.cs.gnscommon.utils.ThreadUtils;

import java.awt.geom.Point2D;
import java.io.File;
import java.io.IOException;
import java.net.InetAddress;
import java.net.URISyntaxException;
import java.net.UnknownHostException;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;

import javax.swing.JDialog;
import javax.swing.JOptionPane;

import org.apache.commons.cli.CommandLine;
import org.apache.commons.cli.CommandLineParser;
import org.apache.commons.cli.GnuParser;
import org.apache.commons.cli.HelpFormatter;
import org.apache.commons.cli.Option;
import org.apache.commons.cli.OptionBuilder;
import org.apache.commons.cli.Options;
import org.apache.commons.cli.ParseException;

/**
 * Starts a set of EC2 instances.
 * Reads from a XML configuration file whose name is specified on the command line.
 *
 * Typical use:
 *
 * java -cp jars/GNS.jar edu.umass.cs.gnsserver.installer.EC2Runner -create dev
 *
 * @author westy
 */
@Deprecated
public class EC2Runner {

    private static final String FILESEPARATOR = System.getProperty("file.separator");
    private static final String CREDENTIALSFILE = System.getProperty("user.home") + FILESEPARATOR + ".aws"
            + FILESEPARATOR + "credentials";
    private static final DataStoreType DEFAULT_DATA_STORE_TYPE = DataStoreType.MONGO;
    private static final AMIRecordType DEFAULT_AMI_RECORD_TYPE = AMIRecordType.Amazon_Linux_AMI_2013_03_1;
    private static final String DEFAULT_EC2_USERNAME = "ec2-user";
    /**
     * Contains information read from config file on what hosts we are trying to start.
     */
    private static List<EC2RegionSpec> regionsList = new ArrayList<EC2RegionSpec>();
    /**
     * Stores information about instances that have started.
     */
    private static ConcurrentHashMap<String, HostInfo> hostTable = new ConcurrentHashMap<String, HostInfo>();
    //
    private static final int STARTINGNODENUMBER = 0;
    private static ConcurrentHashMap<String, String> hostsThatDidNotStart = new ConcurrentHashMap<String, String>();
    private static DataStoreType dataStoreType = DEFAULT_DATA_STORE_TYPE;
    private static AMIRecordType amiRecordType = DEFAULT_AMI_RECORD_TYPE;
    private static String ec2UserName = DEFAULT_EC2_USERNAME;
    private static String configName = null;

    private static void loadConfig(String configName) {
        EC2ConfigParser parser = new EC2ConfigParser(configName);
        for (EC2RegionSpec spec : parser.getRegions()) {
            //System.out.println(spec.toString());
            regionsList.add(new EC2RegionSpec(spec.getRegion(), spec.getCount(), spec.getIp()));
        }
        ec2UserName = parser.getEc2username();
        dataStoreType = parser.getDataStoreType();
        amiRecordType = parser.getAmiRecordType();
    }

    /**
     * Starts a set of EC2 hosts running GNS that we call a runset.
     *
     * @param runSetName
     */
    public static void createRunSetMulti(String runSetName) {
        int timeout = AWSEC2.DEFAULTREACHABILITYWAITTIME;
        System.out.println("EC2 User Name: " + ec2UserName);
        System.out.println("AMI Name: " + amiRecordType.toString());
        System.out.println("Datastore: " + dataStoreType.toString());
        //preferences.put(RUNSETNAME, runSetName); // store the last one
        startAllMonitoringAndGUIProcesses();
        attachShutDownHook(runSetName);
        ArrayList<Thread> threads = new ArrayList<Thread>();
        // use threads to do a bunch of installs in parallel
        do {
            hostsThatDidNotStart.clear();
            //StatusModel.getInstance().queueDeleteAllEntries(); // for gui
            int cnt = STARTINGNODENUMBER;
            for (EC2RegionSpec regionSpec : regionsList) {
                int i;
                for (i = 0; i < regionSpec.getCount(); i++) {
                    threads.add(new EC2RunnerThread(runSetName, regionSpec.getRegion(), Integer.toString(cnt),
                            i == 0 ? regionSpec.getIp() : null, timeout));
                    cnt = cnt + 1;
                }
            }
            for (Thread thread : threads) {
                thread.start();
            }
            // and wait for all of them to complete
            try {
                for (Thread thread : threads) {
                    thread.join();
                }
            } catch (InterruptedException e) {
                System.out.println("Problem joining threads: " + e);
            }

            if (!hostsThatDidNotStart.isEmpty()) {
                System.out.println("Hosts that did not start: " + hostsThatDidNotStart.keySet());
                timeout = (int) ((float) timeout * 1.5);
                System.out.println("Maybe kill them all and try again with timeout " + timeout + "ms?");
                if (showDialog("Hosts that did not start: " + hostsThatDidNotStart.keySet()
                        + "\nKill them all and try again with with timeout " + timeout + "ms?"
                        + "\nIf you don't respond in 10 seconds this will happen.", 10000)) {
                    System.out.println("Yes, kill them all and try again with timeout " + timeout + "ms.");
                    terminateRunSet(runSetName);
                } else {
                    terminateRunSet(runSetName);
                    System.out.println("No, kill them all and quit.");
                    return;
                }
            }

            threads.clear();

            // keep repeating until everything starts
        } while (!hostsThatDidNotStart.isEmpty());

        // got a complete set running... now on to step 2
        System.out.println(hostTable.toString());
        // after we know all the hosts are we run the last part

        System.out.println("Hosts that did not start: " + hostsThatDidNotStart.keySet());
        // write out a config file that the GNS installer can use for this set of EC2 hosts
        writeGNSINstallerConf(configName);
        removeShutDownHook();
        System.out.println("Finished creation of Run Set " + runSetName);
    }

    private static final String keyName = "aws";
    // this one installs mondoDB
    private static final String mongoInstallScript = "#!/bin/bash\n" + "cd /home/ec2-user\n"
            + "yum --quiet --assumeyes update\n" + "yum --quiet --assumeyes install emacs\n" // for debugging
            + "yum --quiet --assumeyes install git\n" + "yum --quiet --assumeyes install java-1.8.0-openjdk-devel\n"
            + "echo \\\"[MongoDB]\n" // crazy double escaping for JAVA and BASH going on here!!
            + "name=MongoDB Repository\n" + "baseurl=http://downloads-distro.mongodb.org/repo/redhat/os/x86_64\n"
            + "gpgcheck=0\n" + "enabled=1\\\" > mongodb.repo\n" // crazy double escaping for JAVA and BASH going on here!!
            + "mv mongodb.repo /etc/yum.repos.d/mongodb.repo\n" + "yum --quiet --assumeyes install mongodb-org\n"
            + "service mongod start\n"
            // fix the sudoers so ssh sudo works all the time
            + "chmod ug+rw /etc/sudoers\n" + "sed -i 's/requiretty/!requiretty/' /etc/sudoers\n";
    private static final String mongoShortInstallScript = "#!/bin/bash\n" + "cd /home/ec2-user\n"
            + "yum --quiet --assumeyes update\n"
            //+ "yum --quiet --assumeyes install emacs\n" // for debugging
            + "service mongod start\n";
    private static final String cassandraInstallScript = "#!/bin/bash\n" + "cd /home/ubuntu\n" //          + "echo \\\"[datastax]\n"
    //          + "name = DataStax Repo for Apache Cassandra\n"
    //          + "baseurl = http://rpm.datastax.com/community\n"
    //          + "enabled = 1\n"
    //          + "gpgcheck = 0\n\\\" > /etc/yum.repos.d/datastax.repo\n"
    //          + "yum --quiet --assumeyes install dsc12\n"
    //          + "cassandra\n"
    ;

    /**
     * This is called to initialize an EC2 host for use as A GNS server in a region. It starts the host, loads all the necessary
     * software and copies the JAR files over. We also collect info about this host, like it's IP address and geographic location.
     * When every host is initialized and we have collected all the IPs, phase two is called.
     *
     * @param region - the EC2 region where we are starting this host
     * @param runSetName - so we can terminate them all together
     * @param id - the GNS ID of this server
     * @param elasticIP
     * @param timeout
     */
    public static void initAndUpdateEC2Host(RegionRecord region, String runSetName, String id, String elasticIP,
            int timeout) {
        String installScript;
        AMIRecord amiRecord = AMIRecord.getAMI(amiRecordType, region);
        if (amiRecord == null) {
            System.out.println("Invalid combination of " + amiRecordType + " and Region " + region.name());
            return;
        }
        switch (dataStoreType) {
        case CASSANDRA:
            installScript = cassandraInstallScript;
            break;
        default: // MONGO
            if (amiRecordType.toString().contains("Amazon_Linux")) {
                installScript = mongoInstallScript;
            } else {
                switch (amiRecordType) {
                case MongoDB_2_4_8_with_1000_IOPS:
                    installScript = mongoShortInstallScript;
                    break;
                case Mongo_2014_5_6:
                    installScript = null;
                    break;
                case Mongo_2014_5_6_micro:
                    installScript = null;
                    break;
                case Mongo_2015_6_25_vpc:
                    installScript = null;
                    break;
                case Mongo_2016_6_16_micro:
                    installScript = null;
                    break;
                default:
                    System.out.println("Invalid combination of " + amiRecordType + " and " + dataStoreType);
                    return;
                }
            }
        }

        String idString = id.toString();
        //    StatusModel.getInstance().queueAddEntry(id); // for the gui
        //    StatusModel.getInstance().queueUpdate(id, region.name() + ": [Unknown hostname]", null, null);
        try {
            AWSCredentials credentials = new PropertiesCredentials(new File(CREDENTIALSFILE));
            //Create Amazon Client object
            AmazonEC2 ec2 = new AmazonEC2Client(credentials);
            String nodeName = "GNS Node " + idString;
            System.out.println("Starting install for " + nodeName + " in " + region.name() + " as part of run set "
                    + runSetName);
            HashMap<String, String> tags = new HashMap<>();
            tags.put("runset", runSetName);
            tags.put("id", idString);
            //      StatusModel.getInstance().queueUpdate(id, "Creating instance");
            // create an instance
            Instance instance = AWSEC2.createAndInitInstance(ec2, region, amiRecord, nodeName, keyName,
                    amiRecord.getSecurityGroup(), installScript, tags, elasticIP, timeout);
            if (instance != null) {
                //        StatusModel.getInstance().queueUpdate(id, "Instance created");
                //        StatusModel.getInstance().queueUpdate(id, StatusEntry.State.INITIALIZING);
                // toString our ip
                String hostname = instance.getPublicDnsName();
                InetAddress inetAddress = InetAddress.getByName(hostname);
                String ip = inetAddress.getHostAddress();
                // and take a guess at the location (lat, long) of this host
                Point2D location = GEOLocator.lookupIPLocation(ip);
                //        StatusModel.getInstance().queueUpdate(id, hostname, ip, location);
                // update our table of instance information
                hostTable.put(id, new HostInfo(id, hostname, location));

                // and we're done
                //        StatusModel.getInstance().queueUpdate(id, "Waiting for other servers");
            } else {
                System.out.println("EC2 Instance " + idString + " in " + region.name() + " did not in start.");
                //        StatusModel.getInstance().queueUpdate(id, StatusEntry.State.ERROR, "Did not start");
                hostsThatDidNotStart.put(id, id);
            }
        } catch (IOException e) {
            System.out.println("Problem creating EC2 instance " + idString + " in " + region.name() + ": " + e);
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            System.out.println("Problem creating EC2 instance " + idString + " in " + region.name() + ": " + e);
            e.printStackTrace();
        }
    }

    //  private static final String MongoRecordsClass = "edu.umass.cs.gnsserver.database.MongoRecords";
    //  private static final String CassandraRecordsClass = "edu.umass.cs.gnsserver.database.CassandraRecords";

    /**
     * Terminates all the hosts in the named run set.
     *
     * @param name
     */
    public static void terminateRunSet(String name) {
        try {
            AWSCredentials credentials = new PropertiesCredentials(new File(CREDENTIALSFILE));
            //Create Amazon Client object
            AmazonEC2 ec2 = new AmazonEC2Client(credentials);
            for (RegionRecord region : RegionRecord.values()) {
                AWSEC2.setRegion(ec2, region);
                for (Instance instance : AWSEC2.getInstances(ec2)) {
                    if (!instance.getState().getName().equals(InstanceStateRecord.TERMINATED.getName())) {
                        String idString = getTagValue(instance, "id");
                        if (name.equals(getTagValue(instance, "runset"))) {
                            if (idString != null) {
                                //                StatusModel.getInstance().queueUpdate(new String(idString), "Terminating");
                            }
                            AWSEC2.terminateInstance(ec2, instance.getInstanceId());
                            if (idString != null) {
                                //                StatusModel.getInstance().queueUpdate(new String(idString), StatusEntry.State.TERMINATED, "");
                            }
                        }
                    }
                }
            }
        } catch (IOException e) {
            System.out.println("Problem terminating EC2 instances: " + e);
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            System.out.println("Problem terminating EC2 instances: " + e);
            e.printStackTrace();
        }
    }

    private static void populateIDTableForRunset(String name) {
        AWSCredentials credentials = null;
        try {
            //
            credentials = new PropertiesCredentials(new File(CREDENTIALSFILE));
        } catch (IOException e) {
            System.out.println("Problem contacting EC2 instances: " + e);
        }
        //Create Amazon Client object
        AmazonEC2 ec2 = new AmazonEC2Client(credentials);
        for (RegionRecord region : RegionRecord.values()) {
            AWSEC2.setRegion(ec2, region);
            System.out.println("Retrieving instance information in " + region.name() + "...");
            for (Instance instance : AWSEC2.getInstances(ec2)) {
                if (!instance.getState().getName().equals(InstanceStateRecord.TERMINATED.getName())) {
                    String idString = getTagValue(instance, "id");
                    if (idString != null && name.equals(getTagValue(instance, "runset"))) {
                        String id = new String(idString);
                        String hostname = instance.getPublicDnsName();
                        String ip = getHostIPSafe(hostname);
                        // and take a guess at the location (lat, long) of this host
                        Point2D location = GEOLocator.lookupIPLocation(ip);
                        hostTable.put(id, new HostInfo(id, hostname, location));
                    }
                }
            }
        }
    }

    private static String getHostIPSafe(String hostname) {
        InetAddress inetAddress;
        try {
            inetAddress = InetAddress.getByName(hostname);
            return inetAddress.getHostAddress();
        } catch (UnknownHostException e) {
            return "Unknown";
        }
    }

    private static void describeRunSet(final String name) {
        populateIDTableForRunset(name);
        for (HostInfo info : hostTable.values()) {
            System.out.println(info);
        }
    }

    private static void writeGNSINstallerConfForRunSet(final String name) {
        populateIDTableForRunset(name);
        for (HostInfo info : hostTable.values()) {
            System.out.println(info);
        }
        writeGNSINstallerConf(configName);
    }

    private static String getTagValue(Instance instance, String key) {
        for (Tag tag : instance.getTags()) {
            if (key.equals(tag.getKey())) {
                return tag.getValue();
            }
        }
        return null;
    }

    // COMMAND LINE STUFF
    private static HelpFormatter formatter = new HelpFormatter();
    private static Options commandLineOptions;

    @SuppressWarnings("static-access")
    private static CommandLine initializeOptions(String[] args) throws ParseException {
        Option help = new Option("help", "Prints Usage");

        Option terminate = OptionBuilder.withArgName("runSet name").hasArg()
                .withDescription("terminate a runset's instances").create("terminate");
        //Option terminateAll = new Option("terminateAll", "terminate all runsets");
        //Option create = new Option("create", "create a runset");
        Option create = OptionBuilder.withArgName("runSet name").hasArg().withDescription("create a runset")
                .create("create");
        Option describe = OptionBuilder.withArgName("runSet name").hasArg().withDescription("describe a runset")
                .create("describe");
        Option writeConfig = OptionBuilder.withArgName("runSet name").hasArg()
                .withDescription("write a GNS Installer config file for runset").create("writeConfig");
        Option dataStore = OptionBuilder.withArgName("data store type").hasArg().withDescription("data store type")
                .create("datastore");

        commandLineOptions = new Options();
        commandLineOptions.addOption(terminate);
        commandLineOptions.addOption(create);
        commandLineOptions.addOption(describe);
        commandLineOptions.addOption(writeConfig);
        commandLineOptions.addOption(dataStore);
        commandLineOptions.addOption(help);

        CommandLineParser parser = new GnuParser();
        return parser.parse(commandLineOptions, args);
    }

    private static void printUsage() {
        formatter.printHelp("java -cp GNS.jar edu.umass.cs.gnsserver.main.EC2Installer <options>",
                commandLineOptions);
    }

    private static void startAllMonitoringAndGUIProcesses() {
        java.awt.EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                //        StatusFrame.getInstance().setVisible(true);
                //        StatusModel.getInstance().addUpdateListener(StatusFrame.getInstance());
                //        MapFrame.getInstance().setVisible(true);
                //        ScreenUtils.putOnWidestScreen(MapFrame.getInstance());
                //        StatusModel.getInstance().addUpdateListener(MapFrame.getInstance());
            }
        });
        //    try {
        ////      new StatusListener().start();
        //    } catch (IOException e) {
        //      System.out.println("Unable to start Status Listener: " + e.getMessage());
        //    }
    }

    private static boolean showDialog(String message, final long timeout) {
        try {
            JOptionPane optionPane = new JOptionPane(message, JOptionPane.QUESTION_MESSAGE,
                    JOptionPane.YES_NO_OPTION);
            final JDialog dlg = optionPane.createDialog("Error");
            new Thread(new Runnable() {
                @Override
                public void run() {
                    {
                        ThreadUtils.sleep(timeout);
                        dlg.dispose();
                    }
                }
            }).start();
            dlg.setVisible(true);
            int value = ((Integer) optionPane.getValue()).intValue();
            if (value == JOptionPane.YES_OPTION) {
                return true;
            } else {
                return false;
            }
        } catch (RuntimeException e) {
            return true;
        }
    }

    private static Thread shutdownHook = null;

    private static void attachShutDownHook(final String runSetName) {
        shutdownHook = new Thread() {
            @Override
            public void run() {
                // Just an extra check to make sure we don't kill a perfectly good run set.
                if (!hostsThatDidNotStart.isEmpty()) {
                    System.out.println("Killing all instances.");
                    terminateRunSet(runSetName);
                }
            }
        };
        Runtime.getRuntime().addShutdownHook(shutdownHook);
        System.out.println("ShutdownHook attached.");
    }

    private static void removeShutDownHook() {
        if (shutdownHook != null) {
            Runtime.getRuntime().removeShutdownHook(shutdownHook);
        }
        shutdownHook = null;
        System.out.println("ShutdownHook removed.");
    }

    /**
     * The main routine.
     *
     * @param args
     */
    public static void main(String[] args) {
        try {
            CommandLine parser = initializeOptions(args);
            if (parser.hasOption("help")) {
                printUsage();
                System.exit(1);
            }
            String createRunsetName = parser.getOptionValue("create");
            String terminateRunsetName = parser.getOptionValue("terminate");
            String runsetDescribe = parser.getOptionValue("describe");
            String runSetWriteConfig = parser.getOptionValue("writeConfig");
            String dataStoreName = parser.getOptionValue("datastore");

            if (dataStoreName != null) {
                try {
                    dataStoreType = DataStoreType.valueOf(dataStoreName);
                } catch (IllegalArgumentException e) {
                    System.out.println("Unknown data store type " + dataStoreName + "; exiting.");
                    System.exit(1);
                }
            }

            configName = createRunsetName != null ? createRunsetName
                    : terminateRunsetName != null ? terminateRunsetName
                            : runsetDescribe != null ? runsetDescribe
                                    : runSetWriteConfig != null ? runSetWriteConfig : null;

            System.out.println("Config name: " + configName);
            if (configName != null) {
                loadConfig(configName);
            }

            if (createRunsetName != null) {
                createRunSetMulti(createRunsetName);
            } else if (terminateRunsetName != null) {
                terminateRunSet(terminateRunsetName);
            } else if (runsetDescribe != null) {
                describeRunSet(runsetDescribe);
            } else if (runSetWriteConfig != null) {
                writeGNSINstallerConfForRunSet(runSetWriteConfig);
            } else {
                printUsage();
                System.exit(1);
            }

        } catch (ParseException e1) {
            e1.printStackTrace();
            printUsage();
            System.exit(1);
        }
        System.exit(0);
    }

    private static class EC2RunnerThread extends Thread {

        String runSetName;
        RegionRecord region;
        String id;
        String ip;
        int timeout;

        public EC2RunnerThread(String runSetName, RegionRecord region, String id, String ip, int timeout) {
            super("Install Start " + id);
            this.runSetName = runSetName;
            this.region = region;
            this.id = id;
            this.ip = ip;
            this.timeout = timeout;
        }

        @Override
        public void run() {
            EC2Runner.initAndUpdateEC2Host(region, runSetName, id, ip, timeout);
        }
    }

    private static void writeGNSINstallerConf(String configName) {
        File jarPath = getJarPath();
        System.out.println("Jar path: " + jarPath);
        String confFileDirectory = jarPath.getParent() + FILESEPARATOR + "conf" + FILESEPARATOR + configName
                + "-init";
        //String confFileLocation = jarPath.getParent() + FILESEPARATOR + "conf" + FILESEPARATOR + "gnsInstaller" + FILESEPARATOR + configName + ".xml";
        WriteConfFile.writeConfFiles(configName, confFileDirectory, keyName, ec2UserName, "linux", "MONGO",
                hostTable);
        //WriteConfFile.writeXMLFile(confFileLocation, keyName, ec2UserName, "linux", dataStoreType.toString(), hostTable);
    }

    private static File getJarPath() {
        try {
            return new File(GNSConfig.class.getProtectionDomain().getCodeSource().getLocation().toURI());
        } catch (URISyntaxException e) {
            GNSConfig.getLogger().info("Unable to get jar location: " + e);
            return null;
        }
    }

}