Package com.sun.grid.installer.gui

Source Code of com.sun.grid.installer.gui.HostPanel

/*___INFO__MARK_BEGIN__*/                   
/*************************************************************************
*
*  The Contents of this file are made available subject to the terms of
*  the Sun Industry Standards Source License Version 1.2
*
*  Sun Microsystems Inc., March, 2001
*
*
*  Sun Industry Standards Source License Version 1.2
*  =================================================
*  The contents of this file are subject to the Sun Industry Standards
*  Source License Version 1.2 (the "License"); You may not use this file
*  except in compliance with the License. You may obtain a copy of the
*  License at http://gridengine.sunsource.net/Gridengine_SISSL_license.html
*
*  Software provided under this License is provided on an "AS IS" basis,
*  WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
*  WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
*  MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
*  See the License for the specific provisions governing your rights and
*  obligations concerning the Software.
*
*  The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
*  Copyright: 2001 by Sun Microsystems, Inc.
*
*  All Rights Reserved.
*
************************************************************************/
/*___INFO__MARK_END__*/
package com.sun.grid.installer.gui;

import com.sun.grid.installer.gui.Host.State;
import com.sun.grid.installer.task.ThreadPoolObserver.ThreadPoolEvent;
import com.sun.grid.installer.task.TestableTask;
import com.sun.grid.installer.task.ValidateHostTask;
import com.sun.grid.installer.task.GetArchitectureTask;
import com.sun.grid.installer.task.InstallTask;
import com.izforge.izpack.installer.InstallData;
import com.izforge.izpack.installer.InstallerFrame;
import com.izforge.izpack.installer.IzPanel;
import com.izforge.izpack.util.Debug;
import com.izforge.izpack.util.VariableSubstitutor;
import com.sun.grid.installer.task.TaskHandler;
import com.sun.grid.installer.task.TaskThreadFactory;
import com.sun.grid.installer.task.ThreadPoolObserver;
import com.sun.grid.installer.task.ThreadPoolObserver.ThreadPoolListener;
import com.sun.grid.installer.util.Config;
import com.sun.grid.installer.util.FileHandler;
import com.sun.grid.installer.util.Util;

import com.sun.grid.installer.util.cmd.CmdExec;
import java.awt.Font;
import java.awt.FontMetrics;
import java.awt.event.ActionListener;
import java.awt.event.FocusAdapter;
import java.awt.event.FocusEvent;
import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Properties;
import java.util.Vector;
import java.util.Set;
import java.util.concurrent.Executors;
import java.util.concurrent.RejectedExecutionException;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.text.MessageFormat;

import java.util.Enumeration;
import java.util.HashMap;
import java.util.Hashtable;
import java.util.Map;
import javax.swing.JButton;
import javax.swing.JFileChooser;
import javax.swing.JOptionPane;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.JTextField;
import javax.swing.SwingUtilities;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ListSelectionEvent;
import javax.swing.event.TableColumnModelEvent;
import javax.swing.event.TableColumnModelListener;
import javax.swing.event.TableModelEvent;
import javax.swing.event.TableModelListener;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.JTableHeader;
import javax.swing.table.TableColumn;
import net.n3.nanoxml.XMLElement;

public class HostPanel extends IzPanel implements Config,
                                                  ThreadPoolListener,
                                                  TaskHandler {

    // Contains the removed listeners had been assigned to next button.
    private ActionListener[] nextButtonListeners = null;

    // Indicate the installation setups
    private boolean isQmasterInst = true;
    private boolean isShadowdInst = true;
    private boolean isExecdInst = true;
    private boolean isBdbInst = true;
    private boolean isExpressInst = true;

    // Indicates whether the error label is visible
    private boolean errorMessageVisible = false;

    // Indicates the actual state of the process
    public static boolean installMode = false;
    public static boolean checkMode = false;

    // Table column headers and their tooltips
    public static String[] SELECTION_TABS = null;
    public static String[] SELECTION_TABS_TOOLTIPS = null;
    public static String[] INSTALL_TABS = null;
    public static String[] INSTALL_TABS_TOOLTIPS = null;
   
    private static Properties localizedMessages = null;

    private Host firstTaskHost, lastTaskHost;

    // Contains the tables(3) shown on the panel
    private Vector<HostTable> tables;

    // Host lists(3) handeled by the tables
    private Vector<HostList> lists;

    // Stores all of the hosts which were attempted to be installed
    // (stored in the success table after host validation)
    private List<Host> validHostList;

    // Thread pools and their observer for tasks
    private ThreadPoolExecutor threadPool = null;
    private ThreadPoolExecutor singleThreadPool = null;
    private ThreadPoolObserver observer = null;

    /** Creates new HostPanel */
    public HostPanel(InstallerFrame parent, InstallData idata) {
        super(parent, idata);
        //TODO: verify that shell name is OK

        init();
    }

    /**
     * Initializes the GUI
     */
    private void init() {
        /**
         * Reinit. Necessary for Start Over feature
         */
        removeAll();

        threadPool = null;
        singleThreadPool = null;
        observer = null;

        isQmasterInst = true;
        isShadowdInst = true;
        isExecdInst = true;
        isBdbInst = true;
        isExpressInst = true;
        errorMessageVisible = false;
        installMode = false;
        checkMode = false;

        localizedMessages = new Properties();
       
        HostConfigFrame.getInstance(parent, idata);
       
        /**
         * Build gui
         */
        SELECTION_TABS = new String[]{
                    getLabel("tab.all.label"),
                    getLabel("tab.reachable.label"),
                    getLabel("tab.unreachable.label")};
        SELECTION_TABS_TOOLTIPS = new String[]{
                    getLabel("tab.all.label.tooltip"),
                    getLabel("tab.reachable.label.tooltip"),
                    getLabel("tab.unreachable.label.tooltip")};
        INSTALL_TABS = new String[]{
                    getLabel("tab.installing.label"),
                    getLabel("tab.done.label"),
                    getLabel("tab.failed.label")};
        INSTALL_TABS_TOOLTIPS = new String[]{
                    getLabel("tab.installing.label.tooltip"),
                    getLabel("tab.done.label.tooltip"),
                    getLabel("tab.failed.label.tooltip")};

        if (idata != null) {
            Set<String> keys = idata.langpack.keySet();
            for (Iterator<String> it = keys.iterator(); it.hasNext();) {
                String key = it.next();
                if (key.startsWith("msg.")) {
                    localizedMessages.put(key, idata.langpack.get(key));
                }
            }
        }

        initComponents();
//        setComponentSelectionVisible(advancedMode);
       
        progressBar.setVisible(false);
        progressBar.setMinimum(0);
        progressBar.setStringPainted(true);
        cancelB.setVisible(false);
        statusBar.setVisible(false);

        observer = new ThreadPoolObserver();
        observer.addThreadPoolListener(this);

        hostTF.addMouseListener(new MouseAdapter() {

            @Override
            public void mouseClicked(MouseEvent e) {
                super.mouseClicked(e);
                if (errorMessageVisible) {
                    statusBar.setVisible(false);
                }
            }
        });

        hostTF.addKeyListener(new KeyAdapter() {

            @Override
            public void keyTyped(KeyEvent e) {
                if (e.getKeyChar() != '\n' && errorMessageVisible) {
                    statusBar.setVisible(false);
                }
            }
        });

        hostRB.addFocusListener(new FocusAdapter() {

            @Override
            public void focusGained(FocusEvent e) {
                if (!hostRB.isSelected() && errorMessageVisible) {
                    statusBar.setVisible(false);
                }
            }
        });

        fileRB.addFocusListener(new FocusAdapter() {

            @Override
            public void focusGained(FocusEvent e) {
                if (!fileRB.isSelected() && errorMessageVisible) {
                    statusBar.setVisible(false);
                }
            }
        });

        String[] selectionHeaders = getSelectionLabelVars();
        String[] headerToltips = new String[selectionHeaders.length];
        for (int i = 0; i < selectionHeaders.length; i++) {
            headerToltips[i] = getTooltip(selectionHeaders[i]);
        }

        lists = new Vector<HostList>();
        tables = new Vector<HostTable>();
        HostList list;
        HostTable table;
        tabbedPane.removeAll();
        for (int i = 0; i < SELECTION_TABS.length; i++) {
            list = new HostList();
            table = new HostTable(this);
            lists.add(list);
            tables.add(table);

            Properties tableLangProps = new Properties();
            tableLangProps.put("msg.qmasterhost.already.selected", getLabel("msg.qmasterhost.already.selected"));
            tableLangProps.put("msg.bdbhost.already.selected", getLabel("msg.bdbhost.already.selected"));
            tableLangProps.put("title.confirmation", getLabel("title.confirmation"));
            tableLangProps.put("msg.qmaster.hastobe.admin.submit", getLabel("msg.qmaster.hastobe.admin.submit"));
            tableLangProps.put("installer.warning", getLabel("installer.warning"));
            table.setModel(new HostSelectionTableModel(table, list, getSelectionHeaders(), getSelectionClassTypes(), tableLangProps));
            //table.setAutoResizeMode(JTable.AUTO_RESIZE_OFF);

            table.setTableHeader(new TooltipTableHeader(table.getColumnModel(), headerToltips));
            JTableHeader header = table.getTableHeader();
            final SortedColumnHeaderRenderer headerRenderer = new SortedColumnHeaderRenderer(header, getImageIcon("columns.sorted.asc"), getImageIcon("columns.sorted.desc"));
            for (int col = 0; col < table.getColumnCount(); col++) {
                table.getColumnModel().getColumn(col).setHeaderRenderer(headerRenderer);
            }

            table.getColumn(getLabel("column.state.label")).setCellRenderer(new StateCellRenderer());

            // 6786350 - gui-installer: when moving columns the sorting stays in the column index not on the column ifself
            table.getColumnModel().addColumnModelListener(new TableColumnModelListener() {
                public void columnAdded(TableColumnModelEvent e) {}
                public void columnRemoved(TableColumnModelEvent e) {}
                public void columnMoved(TableColumnModelEvent e) {
                    headerRenderer.setPressedColumn(e.getToIndex());
                    headerRenderer.setSelectedColumn(e.getToIndex(), headerRenderer.getSelectedColumnState());
                }
                public void columnMarginChanged(ChangeEvent e) {}
                public void columnSelectionChanged(ListSelectionEvent e) {}
            });

            header.addMouseListener(new TableHeaderListener(header, headerRenderer));

            table.getModel().addTableModelListener(new TableModelListener() {
                public void tableChanged(TableModelEvent e) {
                    HostPanel.this.tableChanged();
                }
            });

            // To the success panel add listener which handles install/next button
            // visibility
            if (i == 1) {
                table.getModel().addTableModelListener(new TableModelListener() {
                    public void tableChanged(TableModelEvent e) {
                        // Just handle next button if the success table is empty (hosts have been removed)
                        if (((DefaultTableModel)e.getSource()).getRowCount() == 0) {
                            enableNextButton(false);
                        }
                    }
                });
            }

            tabbedPane.addTab(SELECTION_TABS[i] + " (" + lists.get(i).size() + ")", new JScrollPane(tables.get(i)));
            tabbedPane.setToolTipTextAt(i, SELECTION_TABS_TOOLTIPS[i]);
        }

        hostTF.requestFocus();
        this.repaint();
    }

    /**
     * Returns with the localized header texts for Selection table
     * @return the localized header texts
     */
    private String[] getSelectionHeaders() {
        String[] selectionLabels = getSelectionLabelVars();

        for (int i = 0; i < selectionLabels.length; i++) {
            selectionLabels[i] = getLabel(selectionLabels[i]);
        }

        return selectionLabels;
    }

    /**
     * Returns with the key of the selection table column headers
     * @return The keys of the selection table column headers
     */
    private String[] getSelectionLabelVars() {
        return new String[]{
                    "column.hostname.label",
                    "column.ip.address.label",
                    "column.arch.label",
                    "column.qmaster.label",
                    "column.shadowd.label",
                    "column.execd.label",
                    "column.exec.spool.dir.label",
                    "column.admin.label",
                    "column.submit.label",
                    "column.bdb.label",
                    "column.state.label"};
    }

    /**
     * Returns with the localized header texts for Install table
     * @return the localized header texts
     */
    private String[] getInstallHeaders() {
        String[] installLabels = getInstallLabelVars();

        for (int i = 0; i < installLabels.length; i++) {
            installLabels[i] = getLabel(installLabels[i]);
        }

        return installLabels;
    }

    /**
     * Returns with the key of the install table column headers
     * @return The keys of the install table column headers
     */
    private String[] getInstallLabelVars() {
        return new String[]{
                    "column.component.label",
                    "column.hostname.label",
                    "column.ip.address.label",
                    "column.arch.label",
                    "column.progress.label",
                    "column.log.label"};
    }

    /**
     * Returns with the class types of the selection table columns
     * @return the class types of the selection table columns
     */
    private Class[] getSelectionClassTypes() {
        List<Class> list = new ArrayList<Class>();

        list.add(String.class);
        list.add(String.class);
        list.add(String.class);
        list.add(Boolean.class);
        list.add(Boolean.class);
        list.add(Boolean.class);
        list.add(String.class);
        list.add(Boolean.class);
        list.add(Boolean.class);
        list.add(Boolean.class);
        list.add(Host.State.class);

        Class[] a = new Class[list.size()];
        return list.toArray(a);
    }

    /**
     * Returns with the class types of the install table columns
     * @return the class types of the intall table columns
     */
    private Class[] getInstallClassTypes() {
        List<Class> list = new ArrayList<Class>();

        list.add(String.class);
        list.add(String.class);
        list.add(String.class);
        list.add(String.class);
        list.add(Host.State.class);
        list.add(String.class);

        Class[] a = new Class[list.size()];
        return list.toArray(a);
    }

   /** This method is called from within the constructor to
    * initialize the form.
    * WARNING: Do NOT modify this code. The content of this method is
    * always regenerated by the Form Editor.
    */
    @SuppressWarnings("unchecked")
    // <editor-fold defaultstate="collapsed" desc="Generated Code">//GEN-BEGIN:initComponents
    private void initComponents() {

       hostButtonGroup = new javax.swing.ButtonGroup();
       tabbedPane = new javax.swing.JTabbedPane();
       allHostsScrollPane = new javax.swing.JScrollPane();
       progressBar = new javax.swing.JProgressBar();
       hostRB = new javax.swing.JRadioButton();
       hostTF = new javax.swing.JTextField();
       addB = new javax.swing.JButton();
       fileB = new javax.swing.JButton();
       fileRB = new javax.swing.JRadioButton();
       componentSelectionPanel = new javax.swing.JPanel();
       shadowCB = new javax.swing.JCheckBox();
       execCB = new javax.swing.JCheckBox();
       adminCB = new javax.swing.JCheckBox();
       submitCB = new javax.swing.JCheckBox();
       qmasterCB = new javax.swing.JCheckBox();
       bdbCB = new javax.swing.JCheckBox();
       statusBar = new javax.swing.JLabel();
       cancelB = new javax.swing.JButton();

       hostButtonGroup.add(hostRB);
       hostButtonGroup.add(fileRB);

       allHostsScrollPane.setBackground(new java.awt.Color(255, 255, 255));
       tabbedPane.addTab("All hosts (0)", allHostsScrollPane);

       progressBar.setMaximum(10);
       progressBar.setMinimum(1);

       hostButtonGroup.add(hostRB);
       hostRB.setSelected(true);
       hostRB.setText(getLabel("hostinput.label"));
       hostRB.setToolTipText(getTooltip("hostinput.label.tooltip"));
       hostRB.addFocusListener(new java.awt.event.FocusAdapter() {
           public void focusGained(java.awt.event.FocusEvent evt) {
               hostRBFocusGained(evt);
           }
       });

       hostTF.setText(getLabel("hostinput.value"));
       hostTF.setToolTipText(getTooltip("hostinput.label.tooltip"));
       hostTF.addActionListener(new java.awt.event.ActionListener() {
           public void actionPerformed(java.awt.event.ActionEvent evt) {
               hostTFActionPerformed(evt);
           }
       });
       hostTF.addFocusListener(new java.awt.event.FocusAdapter() {
           public void focusGained(java.awt.event.FocusEvent evt) {
               hostTFFocusGained(evt);
           }
       });

       addB.setText(getLabel("button.add.label"));
       addB.setToolTipText(getTooltip("button.add.label.tooltip"));
       addB.addActionListener(new java.awt.event.ActionListener() {
           public void actionPerformed(java.awt.event.ActionEvent evt) {
               addBActionPerformed(evt);
           }
       });

       fileB.setText(getLabel("UserInputPanel.search.browse"));
       fileB.addActionListener(new java.awt.event.ActionListener() {
           public void actionPerformed(java.awt.event.ActionEvent evt) {
               fileBActionPerformed(evt);
           }
       });
       fileB.addFocusListener(new java.awt.event.FocusAdapter() {
           public void focusGained(java.awt.event.FocusEvent evt) {
               fileBFocusGained(evt);
           }
       });

       hostButtonGroup.add(fileRB);
       fileRB.setText(getLabel("fromfile.label"));
       fileRB.setToolTipText(getTooltip("fromfile.label.tooltip"));
       fileRB.addFocusListener(new java.awt.event.FocusAdapter() {
           public void focusGained(java.awt.event.FocusEvent evt) {
               fileRBFocusGained(evt);
           }
       });

       shadowCB.setText(getLabel("component.shadow.host.label"));
       shadowCB.setToolTipText(getTooltip("component.shadow.host.label.tooltip"));

       execCB.setSelected(true);
       execCB.setText(getLabel("component.execd.host.label"));
       execCB.setToolTipText(getTooltip("component.execd.host.label.tooltip"));

       adminCB.setSelected(true);
       adminCB.setText(getLabel("component.admin.host.label"));
       adminCB.setToolTipText(getTooltip("component.admin.host.label.tooltip"));

       submitCB.setSelected(true);
       submitCB.setText(getLabel("component.submit.host.label"));
       submitCB.setToolTipText(getTooltip("component.submit.host.label.tooltip"));

       qmasterCB.setText(getLabel("component.qmaster.host.label"));
       qmasterCB.setToolTipText(getTooltip("component.shadow.host.label.tooltip"));

       bdbCB.setText(getLabel("component.bdb.host.label"));
       bdbCB.setToolTipText(getTooltip("component.bdb.host.label.tooltip"));

       org.jdesktop.layout.GroupLayout componentSelectionPanelLayout = new org.jdesktop.layout.GroupLayout(componentSelectionPanel);
       componentSelectionPanel.setLayout(componentSelectionPanelLayout);
       componentSelectionPanelLayout.setHorizontalGroup(
           componentSelectionPanelLayout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
           .add(0, 666, Short.MAX_VALUE)
           .add(componentSelectionPanelLayout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
               .add(componentSelectionPanelLayout.createSequentialGroup()
                   .addContainerGap()
                   .add(qmasterCB)
                   .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
                   .add(shadowCB)
                   .addPreferredGap(org.jdesktop.layout.LayoutStyle.UNRELATED)
                   .add(execCB)
                   .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
                   .add(bdbCB)
                   .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
                   .add(adminCB)
                   .addPreferredGap(org.jdesktop.layout.LayoutStyle.UNRELATED)
                   .add(submitCB)
                   .addContainerGap(62, Short.MAX_VALUE)))
       );
       componentSelectionPanelLayout.setVerticalGroup(
           componentSelectionPanelLayout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
           .add(0, 34, Short.MAX_VALUE)
           .add(componentSelectionPanelLayout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
               .add(componentSelectionPanelLayout.createSequentialGroup()
                   .addContainerGap()
                   .add(componentSelectionPanelLayout.createParallelGroup(org.jdesktop.layout.GroupLayout.BASELINE)
                       .add(shadowCB)
                       .add(execCB)
                       .add(qmasterCB)
                       .add(bdbCB)
                       .add(adminCB)
                       .add(submitCB))
                   .addContainerGap(org.jdesktop.layout.GroupLayout.DEFAULT_SIZE, Short.MAX_VALUE)))
       );

       statusBar.setForeground(new java.awt.Color(255, 0, 0));
       statusBar.setIcon(getImageIcon("error.small"));

       cancelB.setText(getLabel("button.cancel.label"));
       cancelB.setToolTipText(getTooltip("button.cancel.label.tooltip"));
       cancelB.addActionListener(new java.awt.event.ActionListener() {
           public void actionPerformed(java.awt.event.ActionEvent evt) {
               cancelBActionPerformed(evt);
           }
       });

       org.jdesktop.layout.GroupLayout layout = new org.jdesktop.layout.GroupLayout(this);
       this.setLayout(layout);
       layout.setHorizontalGroup(
           layout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
           .add(layout.createSequentialGroup()
               .addContainerGap()
               .add(layout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
                   .add(layout.createSequentialGroup()
                       .add(tabbedPane, org.jdesktop.layout.GroupLayout.DEFAULT_SIZE, 1300, Short.MAX_VALUE)
                       .addContainerGap())
                   .add(layout.createSequentialGroup()
                       .add(progressBar, org.jdesktop.layout.GroupLayout.DEFAULT_SIZE, 1184, Short.MAX_VALUE)
                       .add(9, 9, 9)
                       .add(cancelB)
                       .addContainerGap())
                   .add(layout.createSequentialGroup()
                       .add(layout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
                           .add(hostRB)
                           .add(fileRB))
                       .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
                       .add(layout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
                           .add(fileB)
                           .add(layout.createSequentialGroup()
                               .add(hostTF, org.jdesktop.layout.GroupLayout.DEFAULT_SIZE, 815, Short.MAX_VALUE)
                               .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)))
                       .add(addB)
                       .add(284, 284, 284))
                   .add(layout.createSequentialGroup()
                       .add(componentSelectionPanel, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE, org.jdesktop.layout.GroupLayout.DEFAULT_SIZE, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE)
                       .addContainerGap(646, Short.MAX_VALUE))
                   .add(layout.createSequentialGroup()
                       .add(statusBar, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE, 813, Short.MAX_VALUE)
                       .addContainerGap())))
       );
       layout.setVerticalGroup(
           layout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
           .add(org.jdesktop.layout.GroupLayout.TRAILING, layout.createSequentialGroup()
               .add(layout.createParallelGroup(org.jdesktop.layout.GroupLayout.BASELINE)
                   .add(hostRB)
                   .add(addB)
                   .add(hostTF, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE, org.jdesktop.layout.GroupLayout.DEFAULT_SIZE, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE))
               .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
               .add(layout.createParallelGroup(org.jdesktop.layout.GroupLayout.BASELINE)
                   .add(fileRB)
                   .add(fileB))
               .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
               .add(statusBar, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE, 23, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE)
               .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
               .add(componentSelectionPanel, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE, 34, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE)
               .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
               .add(tabbedPane, org.jdesktop.layout.GroupLayout.DEFAULT_SIZE, 330, Short.MAX_VALUE)
               .addPreferredGap(org.jdesktop.layout.LayoutStyle.RELATED)
               .add(layout.createParallelGroup(org.jdesktop.layout.GroupLayout.LEADING)
                   .add(cancelB, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE, 20, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE)
                   .add(progressBar, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE, 20, org.jdesktop.layout.GroupLayout.PREFERRED_SIZE))
               .addContainerGap())
       );
   }// </editor-fold>//GEN-END:initComponents

    @Override
    public void panelActivate() {
        isQmasterInst = isValueEqualsTrue(VAR_INSTALL_QMASTER);
        isShadowdInst = isValueEqualsTrue(VAR_INSTALL_SHADOW);
        isExecdInst = isValueEqualsTrue(VAR_INSTALL_EXECD);
        isBdbInst = isValueEqualsTrue(VAR_INSTALL_BDB) ||
                idata.getVariable(VAR_SPOOLING_METHOD).equals(idata.getVariable(VAR_SPOOLING_METHOD_BERKELEYDBSERVER));
        isExpressInst = idata.getVariable(VAR_INSTALL_MODE).equals(idata.getVariable(VAR_INSTALL_MODE_EXPRESS));

        //Disable only is we have no reachable host!
        if (((HostSelectionTableModel)tables.get(1).getModel()).getRowCount() == 0) {
           enableNextButton(false);
        }

        triggerInstallButton(true);
        setupComponentSelectionPanel();
        setColumnsWidth();

        if (threadPool == null) {
            threadPool = (ThreadPoolExecutor) Executors.newFixedThreadPool(Util.RESOLVE_THREAD_POOL_SIZE);
            threadPool.setThreadFactory(new TaskThreadFactory());
            observer.setThreadPoolExecutors(threadPool);
        }

        VariableSubstitutor vs = new VariableSubstitutor(idata.getVariables());

        // In some cases the execd spool dir does not get substituted
        idata.setVariable(VAR_EXECD_SPOOL_DIR, vs.substituteMultiple(idata.getVariable(VAR_EXECD_SPOOL_DIR), null));

        // cfg.db.spooling.dir
        String spoolingDir = "";
        if (idata.getVariable(VAR_SPOOLING_METHOD).equals("berkeleydb")) {
            spoolingDir = idata.getVariable(VAR_DB_SPOOLING_DIR_BDB);
        } else {
            spoolingDir = idata.getVariable(VAR_DB_SPOOLING_DIR_BDBSERVER);
        }
        idata.setVariable(VAR_DB_SPOOLING_DIR, vs.substituteMultiple(spoolingDir, null));

        // Substitute the variables in Host.State tooltips
        String key = "";
        String value = "";
        Host.State.localizedTexts.clear();
       
        if (idata.langpack != null) {
            if (isExpressInst) {
                idata.setVariable("tooltip.hint", idata.langpack.getString("express.mode.tooltip.hint"));
            } else {
                idata.setVariable("tooltip.hint", "");
            }
           
            Set<String> keys = idata.langpack.keySet();
            for (Iterator<String> it = keys.iterator(); it.hasNext();) {
                key = it.next();
                value = vs.substituteMultiple((String) idata.langpack.get(key), null);
                if (key.startsWith(LANGID_PREFIX_STATE)) {
                    Host.State.localizedTexts.put(key, value);
                }
            }

            // override the tooltip for perm_execd_spool_dir state dependig in the mode
            if (isExpressInst) {
                value = vs.substituteMultiple((String) idata.langpack.get("state.perm_execd_spool_dir.tooltip.global"), null);
            } else {
                value = vs.substituteMultiple((String) idata.langpack.get("state.perm_execd_spool_dir.tooltip.local"), null);
            }
            Host.State.localizedTexts.put("state.perm_execd_spool_dir.tooltip", value);
        }

        // Initialize the table(s) with the qmaster and the Berkeley DB host if it's necessary
        if (true) {
            try {
                boolean isQmasterExist = HostSelectionTableModel.getQmasterHost() != null;
                boolean isBdbExist = HostSelectionTableModel.getBdbHost() != null;
                enablePrevButton(false);

                ArrayList<String> host = new ArrayList<String>();
                Properties prop = new Properties();
                String arch = "";
                String message = "";
                if (isQmasterInst && !isQmasterExist) {
                    if (!idata.getVariable(VAR_QMASTER_HOST).equals("")) {
                        host.add(idata.getVariable(VAR_QMASTER_HOST));

                        resolveHosts(Host.Type.HOSTNAME, host, true, false, (isShadowdInst == true) ? true : false, false, true, true, idata.getVariable(VAR_EXECD_SPOOL_DIR));

                        if (HostSelectionTableModel.getQmasterHost() == null) {
                            try {
                                arch = lists.get(0).get(0).getArchitecture();
                            } finally {
                                prop.setProperty(PARAMETER_1, arch);
                            }

                            if (!arch.equals("")) {
                                vs = new VariableSubstitutor(prop);
                                message = vs.substituteMultiple(idata.langpack.getString("warning.qmaster.arch.unsupported.message"), null);
                                emitWarning(idata.langpack.getString("installer.warning"), message);
                            }
                        }
                    }
                }
                host.clear();

                if (isBdbInst && !isBdbExist) {
                    if (idata.getVariable(VAR_DB_SPOOLING_SERVER).equals("")) {
                        host.add(Host.localHostName); // TODO localhost or qmaster host ?
                    } else {
                        host.add(idata.getVariable(VAR_DB_SPOOLING_SERVER));
                    }
                    resolveHosts(Host.Type.HOSTNAME, host, false, true, false, false, false, false, idata.getVariable(VAR_EXECD_SPOOL_DIR));

                    if (HostSelectionTableModel.getBdbHost() == null) {
                        try {
                            arch = lists.get(0).get(0).getArchitecture();
                        } finally {
                            prop.setProperty(PARAMETER_1, arch);
                        }

                        if (!arch.equals("")) {
                            vs = new VariableSubstitutor(prop);
                            message = vs.substituteMultiple(idata.langpack.getString("warning.bdbserver.arch.unsupported.message"), null);
                            emitWarning(idata.langpack.getString("installer.warning"), message);
                        }
                    }
                }

                if (tables.get(1).getRowCount() > 0) {
                    enableNextButton(true);
                }
            } catch (Exception e) {
                Debug.error(e);
            } finally {
                vs = null;
                enablePrevButton(true);
            }
        }
    }

    @Override
    public void panelDeactivate() {
        // Don't do anything more if we are after the installation
        if (installMode) return;

        enableNextButton(true);
        triggerInstallButton(false);

        //Set the qmaster and bdb host values
        String qmasterHostName = "";
        if (HostSelectionTableModel.getQmasterHost() != null) {
            qmasterHostName = HostSelectionTableModel.getQmasterHost().getHostname();
        }
        idata.setVariable(VAR_QMASTER_HOST, qmasterHostName);
        String bdbHostName = "";
        if (HostSelectionTableModel.getBdbHost() != null) {
            bdbHostName = HostSelectionTableModel.getBdbHost().getHostname();
        }
        idata.setVariable(VAR_DB_SPOOLING_SERVER, bdbHostName);
    }

    /**
     * Switches between the Next and Install button
     * @param show if true the Install button will be shown, otherwise the Next button.
     */
    private synchronized void triggerInstallButton(boolean show) {
        if (parent == null) {
            return;
        }
       
        JButton nextButton = parent.getNextButton();

        if (show) {
            nextButton.setText(getLabel("InstallPanel.install"));
            nextButton.setIcon(getImageIcon("install"));
            nextButtonListeners = Util.removeListeners(nextButton);

            nextButton.addActionListener(new java.awt.event.ActionListener() {

                public void actionPerformed(java.awt.event.ActionEvent evt) {
                    installButtonActionPerformed();
                }
            });
        } else {
            nextButton.setText(getLabel("installer.next"));
            nextButton.setIcon(getImageIcon("stepforward"));
            Util.removeListeners(nextButton);

            for (int i = 0; i < nextButtonListeners.length; i++) {
                nextButton.addActionListener(nextButtonListeners[i]);
            }
        }
    }

    /**
     * Sets the install/next button's enabled property
     * @param b if true button will be enabled else the button will be disabled
     */
    private synchronized void enableNextButton(boolean enabled) {
        if (parent == null) {
            return;
        }

        JButton nextButton = parent.getNextButton();

        if (enabled && !nextButton.isEnabled()) {
            SwingUtilities.invokeLater(new Runnable(){
                public void run() {
                    parent.unlockNextButton();
                }
            });
        } else if (!enabled && nextButton.isEnabled()) {
            SwingUtilities.invokeLater(new Runnable(){
                public void run() {
                    parent.lockNextButton();
                }
            });
        }
    }

    /**
     * Sets the previous button's enabled property
     * @param b if true button will be enabled else the button will be disabled
     */
    private synchronized void enablePrevButton(boolean enabled) {
        if (parent == null) {
            return;
        }

        if (enabled) {
            SwingUtilities.invokeLater(new Runnable(){
                public void run() {
                    parent.unlockPrevButton();
                }
            });
        } else {
            SwingUtilities.invokeLater(new Runnable(){
                public void run() {
                    parent.lockPrevButton();
                }
            });
        }
    }

    /**
     * Sets both previous and next buttons enabled property
     * @param b if true buttons will be enabled else the buttons will be disabled
     */
    private synchronized void enableBrowseButtons(boolean enabled) {
        enableNextButton(enabled);
        enablePrevButton(enabled);
    }

    /**
     * Returns with localized tooltip for the given key
     * @param key The key which identifies the localized tooltip
     * @return The localized tooltip if there is any. Empty string otherwise.
     */
    public String getTooltip(String key) {
        if (idata == null) {
            return null;
        }

        if (!key.endsWith(TOOLTIP)) {
            key = key + "." + TOOLTIP;
        }

        String tooltip = getString(key);

        VariableSubstitutor vs = new VariableSubstitutor(idata.getVariables());
        tooltip = vs.substituteMultiple(tooltip, null);

        if (tooltip.equals(key)) {
            tooltip = null;
        }

        return tooltip;
    }

    /**
     * Returns with localized text for the given key
     * @param key The key which identifies the localized text
     * @return The localized text if there is any. Empty string otherwise.
     */
    public String getLabel(String key) {
        String label = "";

        label = getString(key);

        return label;
    }

    public boolean isValueEqualsTrue(String key) {
        return idata.getVariable(key).equalsIgnoreCase("true");
    }

    /**
     * Sets the visibility on the Component Selection panel regarding to component
     * selection and instalall mode
     */
    private void setupComponentSelectionPanel() {
        SwingUtilities.invokeLater(new Runnable() {

            public void run() {
                componentSelectionPanel.setVisible(!isExpressInst);
                if (!isExecdInst) {
                    execCB.setSelected(false);
                    execCB.setVisible(false);
                } else {
                    execCB.setSelected(true);
                    execCB.setVisible(true);
                }
                // Shadow component is not default on
                if (!isShadowdInst) {
                   shadowCB.setSelected(false);
                   shadowCB.setVisible(false);
                } else {
                   if (!isQmasterInst || !isExecdInst) {
                      //Enabled if shandalone installation or shadowd+bdb
                      shadowCB.setSelected(true);
                   } else {
                      shadowCB.setSelected(false);
                   }
                   shadowCB.setVisible(true);
                }
                qmasterCB.setSelected(false);
                qmasterCB.setVisible(false);
                bdbCB.setSelected(false);
                bdbCB.setVisible(false);

                // Admin component is not default on
                adminCB.setSelected(false);
            }
        });
    }

    /**
     * Sets the column widths in the tables regards to the component selection and
     * instalall mode
     */
    private void setColumnsWidth() {
        SwingUtilities.invokeLater(new Runnable() {

            public void run() {
                int minWidth;
                TableColumn column;

                for (HostTable table : tables) {
                    try {
                        for (int col = 0; col < table.getColumnModel().getColumnCount() - 1; col++) {
                            column = table.getColumnModel().getColumn(col);
                            String header = (String) column.getIdentifier();
                            minWidth = getColumnTextWidth(table, header) + 30;

                            if (header.equals(getLabel("column.qmaster.label"))) {
                                if (!isQmasterInst) {
                                    column.setPreferredWidth(0);
                                    column.setMinWidth(0);
                                    column.setMaxWidth(0);
                                } else {
                                    column.setMinWidth(minWidth);
                                    column.setPreferredWidth(minWidth);
                                    column.setMaxWidth(minWidth);
                                }
                            } else if (header.equals(getLabel("column.shadowd.label"))) {
                                if (!isShadowdInst) {
                                    column.setPreferredWidth(0);
                                    column.setMinWidth(0);
                                    column.setMaxWidth(0);
                                } else {
                                    column.setMinWidth(minWidth);
                                    column.setPreferredWidth(minWidth);
                                    column.setMaxWidth(minWidth);
                                }
                            } else if (header.equals(getLabel("column.execd.label"))) {
                                if (!isExecdInst) {
                                    column.setPreferredWidth(0);
                                    column.setMinWidth(0);
                                    column.setMaxWidth(0);
                                } else {
                                    column.setMinWidth(minWidth);
                                    column.setPreferredWidth(minWidth);
                                    column.setMaxWidth(minWidth);
                                }
                            } else if (header.equals(getLabel("column.exec.spool.dir.label"))) {
                                // Never show local execd spool dir column!
//                                if (isExecdInst && !isExpressInst) {
//                                    column.setMinWidth(15);
//                                    column.setMaxWidth(Integer.MAX_VALUE);
//                                    column.setPreferredWidth(minWidth);
//                                    column.setWidth(minWidth);
//                                } else {
                                    column.setPreferredWidth(0);
                                    column.setMinWidth(0);
                                    column.setMaxWidth(0);
//                                }
                            } else if (header.equals(getLabel("column.admin.label"))) {
                                column.setMinWidth(minWidth);
                                column.setPreferredWidth(minWidth);
                                column.setMaxWidth(minWidth);
                            } else if (header.equals(getLabel("column.submit.label"))) {
                                column.setMinWidth(minWidth);
                                column.setPreferredWidth(minWidth);
                                column.setMaxWidth(minWidth);
                            } else if (header.equals(getLabel("column.bdb.label"))) {
                                if (!isBdbInst) {
                                    column.setPreferredWidth(0);
                                    column.setMinWidth(0);
                                    column.setMaxWidth(0);
                                } else {
                                    column.setMinWidth(minWidth);
                                    column.setPreferredWidth(minWidth);
                                    column.setMaxWidth(minWidth);
                                }
                            } else {
                                column.setMinWidth(minWidth);
                                column.setPreferredWidth(minWidth);
                                column.setMaxWidth(Integer.MAX_VALUE);
                                column.setWidth(minWidth);
                            }
                        }

                    } catch (IllegalArgumentException e) {
                    }
                }
            }
        });
    }

    /**
     * Returns with the minimum cloumns width for the given header text
     * @param table The table which contains the clomn
     * @param columnHeader The header text of the column
     * @return The minimum column with for the given header text
     */
    private int getColumnTextWidth(JTable table, String columnHeader) {
        JButton defButton = new JButton(columnHeader);
        Font font = table.getTableHeader().getFont();
        FontMetrics metrics = defButton.getFontMetrics(font);

        return SwingUtilities.computeStringWidth(metrics, columnHeader);
    }

    private void addHostsFromTF() {
        try {
            final String pattern = hostTF.getText().trim();
            Host.Type type = Util.isIpPattern(pattern) ? Host.Type.IP : Host.Type.HOSTNAME;
            List<String> list = Util.parsePattern(pattern, type);
            Util.validateHostIDList(list, type);
            resolveHosts(type, list, qmasterCB.isSelected(), bdbCB.isSelected(), shadowCB.isSelected(), execCB.isSelected(), adminCB.isSelected(), submitCB.isSelected(), idata.getVariable(VAR_EXECD_SPOOL_DIR));
        } catch (IllegalArgumentException ex) {
            statusBar.setVisible(true);
            statusBar.setText("Error: " + ex.getMessage());
            hostTF.setSelectionStart(0);
            hostTF.setSelectionEnd(hostTF.getText().length());
            errorMessageVisible = true;
        }
    }

    private void hostTFActionPerformed(java.awt.event.ActionEvent evt) {
        addHostsFromTF();
    }

    private void hostTFFocusGained(java.awt.event.FocusEvent evt) {
        hostRB.setSelected(true);
        hostTF.setEditable(true);
        //addHostFocusGained();
        selectTextField(hostTF);
    }

    private void fileBFocusGained(java.awt.event.FocusEvent evt) {
        fileRB.setSelected(true);
        hostTF.setEditable(false);
    }

    private void fileBActionPerformed(java.awt.event.ActionEvent evt) {
        final JFileChooser fc = new JFileChooser();
        int ret = fc.showOpenDialog(this.getParent());
        if (ret == JFileChooser.APPROVE_OPTION) {
            File f = fc.getSelectedFile();
            try {
                List<String> entryList = FileHandler.readFileContent(f.getAbsolutePath(), false);
                List<Host> allHosts = new ArrayList<Host>(entryList.size());
                for (String entry : entryList) {
//                    try {
                        List<Host> hostsPart = Host.fromStringInstance(entry);

                        // if there are no arguments defined set the default values
                        if (entry.split(Host.SEPARATOR).length == 1) {
                            for (Host h : hostsPart) {
                                h.setQmasterHost(qmasterCB.isSelected());
                                h.setBdbHost(bdbCB.isSelected());
                                h.setShadowHost(shadowCB.isSelected());
                                h.setExecutionHost(execCB.isSelected());
                                h.setAdminHost(adminCB.isSelected());
                                h.setSubmitHost(submitCB.isSelected());
                                h.setSpoolDir(idata.getVariable(VAR_EXECD_SPOOL_DIR));
                            }
                        }

                        allHosts.addAll(hostsPart);
//                    } catch (IllegalArgumentException e) {
//                        Debug.error("Can not parse line '" + entry + "'! " + e);
//                    }
                }
               
                resolveHosts(allHosts);

            } catch (IllegalArgumentException ex) {
                statusBar.setVisible(true);
                statusBar.setText("Error: " + ex.getMessage());
                errorMessageVisible = true;
            } catch (FileNotFoundException ex) {
                statusBar.setVisible(true);
                statusBar.setText("Error: cannot find file: " + f.getAbsolutePath());
                errorMessageVisible = true;
            } catch (IOException ex) {
                statusBar.setVisible(true);
                statusBar.setText("Error: cannot read file: " + f.getAbsolutePath());
                errorMessageVisible = true;
            }
        }
    }

    private void selectTextField(JTextField tf) {
        tf.requestFocus();
        tf.setSelectionStart(0);
        tf.setSelectionEnd(tf.getText().length());
        lastSelectedTF = tf;
    }

    private void hostRBFocusGained(java.awt.event.FocusEvent evt) {
        hostTF.setEditable(true);
        selectTextField(hostTF);
    }

    private void fileRBFocusGained(java.awt.event.FocusEvent evt) {
        hostTF.setEditable(false);
        lastSelectedTF.setSelectionEnd(0);
    }

    private void buttonActionPerformed(java.awt.event.ActionEvent evt) {
        if (hostRB.isSelected()) {
            addHostsFromTF();
            hostRB.requestFocus();
        }
    }

    private void addBActionPerformed(java.awt.event.ActionEvent evt) {
        buttonActionPerformed(evt);
    }

    private void cancelBActionPerformed(java.awt.event.ActionEvent evt) {                                       
        cancelActions();
    }                                      

    /**
     * Cancels the running tasks
     */
    private void cancelActions() {
        ThreadPoolExecutor[] tpes = new ThreadPoolExecutor[]{singleThreadPool, threadPool};

        for (ThreadPoolExecutor tpe : tpes) {
            if (tpe == null) {
                continue;
            }

            //Cancel host selection thread pool
            List<Runnable> waitingTasks = tpe.shutdownNow();

            try {
                tpe.awaitTermination(200, TimeUnit.MILLISECONDS);
                tpe.purge();
            } catch (InterruptedException e) {
                Debug.error(e);
            }

            try {
                if (waitingTasks.size() > 0) {
                    ThreadPoolExecutor tmp = (ThreadPoolExecutor) Executors.newFixedThreadPool(waitingTasks.size() * 2);
                    tmp.setThreadFactory(new TaskThreadFactory());

                    for (Iterator<Runnable> it = waitingTasks.iterator(); it.hasNext();) {
                        TestableTask runnable = (TestableTask) it.next();
                        runnable.setTestMode(true);
                        runnable.setTestExitValue(EXIT_VAL_CMDEXEC_INTERRUPTED);
                        runnable.setTestOutput(new Vector<String>());
                        try {
                            Debug.trace("Cancel task: " + runnable.getTaskName());
                            tmp.execute(runnable);
                            tmp.remove(runnable);
                        } catch (Exception e) {
                            Debug.error("Failed to cancel task: " + runnable.getTaskName());
                        }
                    }
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }

        progressBar.setVisible(false);
        cancelB.setVisible(false);
    }

    /**
     * Sets the tabbed panes title depending on the
     */
    public synchronized void tableChanged() {
        String[] tabTitles = (installMode ? HostPanel.INSTALL_TABS : HostPanel.SELECTION_TABS);
        for (int i = 0; i < tables.size(); i++) {
            try {
                tabbedPane.setTitleAt(i, tabTitles[i] + " (" + tables.get(i).getRowCount() + ")");
            } catch (Exception e) {}
       }
    }

    public void addThreadPoolListener(ThreadPoolListener threadPoolListener) {
        observer.addThreadPoolListener(threadPoolListener);
    }

    public void removeThreadPoolListener(ThreadPoolListener threadPoolListener) {
        observer.removeThreadPoolListener(threadPoolListener);
    }

    /**
     * Method to capture {@link ThreadPoolEvent}s
     * @param threadPoolEvent The {@link ThreadPoolEvent} has been fired.
     */
    public void threadPoolActionPerformed(ThreadPoolEvent threadPoolEvent) {
        final int type = threadPoolEvent.getType();
        final ThreadPoolObserver obs = (ThreadPoolObserver) threadPoolEvent.getSource();

        SwingUtilities.invokeLater(new Runnable() {

            public void run() {

                /**
                 * Update main progress bar
                 */
                int cur = obs.getLastRunCompletedTaskCount();
                int max = obs.getLastRunTaskCount();

                String text = getLabel("progressbar.resolving.label");
                if (checkMode) {
                    text = getLabel("progressbar.checking.label");
                } else if (installMode) {
                    text = getLabel("progressbar.installing.label");
                }

                progressBar.setValue(cur);
                progressBar.setMaximum(max);
                progressBar.setString(text + " " + cur + " / " + max);

                switch (type) {
                    case ThreadPoolEvent.EVENT_THREAD_POOL_STARTED: {
                        progressBar.setVisible(true);
                        cancelB.setVisible(true);
                        break;
                    }
                    case ThreadPoolEvent.EVENT_THREAD_POOL_UPDATED: {
                        break;
                    }
                    case ThreadPoolEvent.EVENT_THREAD_POOL_FINISHED: {
                        progressBar.setVisible(false);
                        cancelB.setVisible(false);

                        if (installMode) { // install
                            //if ((tables.get(0).getRowCount() == 0)) { Not reliable!
                                enableNextButton(true);
                                triggerInstallButton(false);
                            //}
                        } else if (checkMode){ // validate
                            // validateHostsAndInstall method takes care about
                            // browse button handling
                        } else if (tables.get(1).getRowCount() > 0) { // resolve
                            enableNextButton(true);
                        }
                        break;
                    }
                }
            }
        });
    }

    /**
     * Resolves the given hosts
     * @param type The {@link Host.Type} type of the host ID list: HOSTNAME or IP
     * @param list The host ID list
     * @param isQmasterHost Indicates whether the host is a qmaster host
     * @param isBDBHost Indicates whether the host is a BDB host
     * @param isShadowHost Indicates whether the host is a shadow host
     * @param isExecutionHost Indicates whether the host is a execution host
     * @param isAdminHost  Indicates whether the host is a admin host
     * @param isSubmitHost Indicates whether the host is a submit host
     * @param execdSpoolDir The execution spool dir for the host
     */
    public void resolveHosts(Host.Type type, List<String> list, boolean isQmasterHost, boolean isBDBHost, boolean isShadowHost, boolean isExecutionHost, boolean isAdminHost, boolean isSubmitHost, String execdSpoolDir) {
        Host h;
        List<Host> hosts = new ArrayList<Host>(list.size());
        for (String hostname : list) {
            h = new Host(type, hostname, isQmasterHost, isBDBHost, isShadowHost, isExecutionHost, isAdminHost, isSubmitHost, execdSpoolDir, Host.State.NEW_UNKNOWN_HOST);

            h.setJvmLibPath(idata.getVariable(VAR_JVM_LIB_PATH));
            h.setJvmAddArgs(idata.getVariable(VAR_ADDITIONAL_JVM_ARGS));
            h.setConnectUser(Util.DEF_CONNECT_USER);
            h.setResolveTimeout(Util.DEF_RESOLVE_TIMEOUT);
            h.setInstallTimeout(Util.DEF_INSTALL_TIMEOUT);
           
            hosts.add(h);
        }

        resolveHosts(hosts);
    }

    /**
     * Resolves the given hosts
     * @param hosts The host list
     */
    public void resolveHosts(List<Host> hosts) {
        HostTable table = tables.get(0);
        HostSelectionTableModel model = (HostSelectionTableModel) table.getModel();

        if (threadPool.isShutdown()) {
            threadPool = (ThreadPoolExecutor) Executors.newFixedThreadPool(Util.RESOLVE_THREAD_POOL_SIZE);
            threadPool.setThreadFactory(new TaskThreadFactory());

            observer.setThreadPoolExecutors(threadPool);
        }

        observer.observe();
       
        for (Host host : hosts) {
            if (host.getConnectUser().equals("")) host.setConnectUser(Util.DEF_CONNECT_USER);
            if (host.getJvmLibPath().equals("")) host.setJvmLibPath(idata.getVariable(VAR_JVM_LIB_PATH));
            if (host.getJvmAddArgs().equals("")) host.setJvmAddArgs(idata.getVariable(VAR_ADDITIONAL_JVM_ARGS));
            if (host.getResolveTimeout() == 0) host.setResolveTimeout(Util.DEF_RESOLVE_TIMEOUT);
            if (host.getInstallTimeout() == 0) host.setInstallTimeout(Util.DEF_INSTALL_TIMEOUT);

            host = model.addHost(host);

            try {
                threadPool.execute(new GetArchitectureTask(host, this, idata.getVariable(VAR_SHELL_NAME), idata.getVariable(VAR_SGE_ROOT),
                        idata.getVariable(VAR_SGE_QMASTER_PORT), idata.getVariable(VAR_EXECD_SPOOL_DIR)));
            } catch (RejectedExecutionException e) {
                setHostState(host, State.CANCELED);
            }
        }
    }

    /**
     * Makes some fixes on the variables. Should be used right before the installation.
     */
    private void fixVariables() {
        VariableSubstitutor vs = new VariableSubstitutor(idata.getVariables());
        String variable = "";

        // cfg.spooling.method.berkeleydbserver
        variable = idata.getVariable(VAR_SPOOLING_METHOD_BERKELEYDBSERVER);
        if (variable.equals("none")) {
            idata.setVariable(VAR_SPOOLING_METHOD_BERKELEYDBSERVER, "berkeleydb");
        }

        // cfg.db.spooling.server
        idata.setVariable(VAR_DB_SPOOLING_SERVER, "none");

        // add.qmaster.host
        if (isQmasterInst) {
            idata.setVariable(VAR_QMASTER_HOST, "none");
        }

        boolean isVarWinSuppSet = false;
        HostList list = lists.get(1);
        Host h = null;
        for (int i = 0; i < list.size(); i++) {
            h = list.get(i);
            // cfg.windows.support
            if (!isVarWinSuppSet && h.getArchitecture().startsWith("win")) {
                idata.setVariable(VAR_WINDOWS_SUPPORT, "true");
                isVarWinSuppSet = true;
            }

            // cfg.hostname.resolving
//            if (!isVarHostResSet) {
//                if (!tmpDomainName.equals("") && !tmpDomainName.equals(Util.getDomainName(h.getHostname()))) {
//                    idata.setVariable(VAR_HOSTNAME_RESOLVING, "false");
//                    isVarHostResSet = true;
//                }
//
//                tmpDomainName = Util.getDomainName(h.getHostname());
//            }

            // cfg.db.spooling.server
            if (h.isBdbHost()) {
                idata.setVariable(VAR_DB_SPOOLING_SERVER, h.getHostname());
            }

            // add.qmaster.host
            if (h.isQmasterHost()) {
                idata.setVariable(VAR_QMASTER_HOST, h.getHostname());
            }
        }
    }

    /**
     *  Sets install data variables depending on the current status of the panel.
     *  Use before filling up the readme file.
     */
    private void createReport() {
        String qmaster = "";
        String execds  = "";
        String shadows = "";
        String admins  = "";
        String submits = "";
        String bdb     = "";

        /**
         * Create lists from succeded hosts
         */
        HostList hostList = tables.get(1).getHostList();
        qmaster = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.qmaster), " ");
        execds = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.execd), " ");
        shadows = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.shadow), " ");
        admins = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.admin), " ");
        submits = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.submit), " ");
        bdb = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.bdb), " ");

        idata.setVariable(VAR_QMASTER_HOST, qmaster);
        idata.setVariable(VAR_EXEC_HOST_LIST, execds);
        idata.setVariable(VAR_SHADOW_HOST_LIST, shadows);
        idata.setVariable(VAR_ADMIN_HOST_LIST, admins);
        idata.setVariable(VAR_SUBMIT_HOST_LIST, submits);
        idata.setVariable(VAR_DB_SPOOLING_SERVER, bdb);

        /**
         * Create lists from failed hosts
         */
        hostList = tables.get(2).getHostList();
        qmaster = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.qmaster), " ");
        execds = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.execd), " ");
        shadows = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.shadow), " ");
        admins = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.admin), " ");
        submits = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.submit), " ");
        bdb = Util.getHostNames(Util.getHosts(hostList, Util.SgeComponents.bdb), " ");

        // remove every host from success admin/submit list if the last task fails
        for (Host h : hostList) {
            if (h.isLastTask()) {
                idata.setVariable(VAR_ADMIN_HOST_LIST, "");
                idata.setVariable(VAR_SUBMIT_HOST_LIST, "");
                break;
            }
        }

        idata.setVariable(VAR_QMASTER_HOST_FAILED, qmaster);
        idata.setVariable(VAR_EXEC_HOST_LIST_FAILED, execds);
        idata.setVariable(VAR_SHADOW_HOST_LIST_FAILED, shadows);
        idata.setVariable(VAR_ADMIN_HOST_LIST_FAILED, admins);
        idata.setVariable(VAR_SUBMIT_HOST_LIST_FAILED, submits);
        idata.setVariable(VAR_DB_SPOOLING_SERVER_FAILED, bdb);

        idata.setVariable("add.sge.jmx.with.ssl", Boolean.toString(isValueEqualsTrue(VAR_SGE_JMX) && isValueEqualsTrue(VAR_JMX_SSL)));

        idata.setVariable("installed.solaris.smf", "false");
        if (isValueEqualsTrue(idata.getVariable(VAR_SGE_ENABLE_SMF))) {
            for (Host host : tables.get(1).getHostList()) {
                if (host.getArchitecture().startsWith("sol-")) {
                    idata.setVariable("installed.solaris.smf", "true");
                }
            }
        }
    }

    /**
     * Sets the enabled property of the host addition controls on the top
     * @param b indicates whether the controls should be disabled.
     */
    private void disableControls(final boolean b) {
        SwingUtilities.invokeLater(new Runnable() {

            public void run() {
                hostRB.setEnabled(!b);
                hostTF.setEnabled(!b);
                fileRB.setEnabled(!b);
                fileB.setEnabled(!b);
                shadowCB.setEnabled(!b);
                execCB.setEnabled(!b);
                adminCB.setEnabled(!b);
                submitCB.setEnabled(!b);
                addB.setEnabled(!b);
                qmasterCB.setEnabled(!b);
                bdbCB.setEnabled(!b);
            }
        });
    }

    /**
     * Clears the selections in all of the tables in order to apply modifications
     */
    private void clearTableSelections() {
        for (JTable table : tables) {
            table.getSelectionModel().clearSelection();
        }
    }

    /**
     * Sets the tables enabled property
     * @param enabled if true the tables will be enabled, otherwise the tables will
     * be disbaled.
     */
    private void setTablesEnabled(final boolean enabled) {
        SwingUtilities.invokeLater(new Runnable() {

            public void run() {
                for (JTable table : tables) {
                    table.setEnabled(enabled);
                }
            }
        });
    }

    /**
     * Sets the sate of the given host
     * @param host The host
     * @param state The state to set
     */
    public void setHostState(Host host, State state) {
           
        /**
         * Set host state on the table(s)
         */
        ((HostTableModel)tables.get(0).getModel()).setHostState(host, state);

        if (checkMode) {
            ((HostTableModel)tables.get(1).getModel()).setHostState(host, state);
        }

        int targetTable = -1;
        switch (state) {
            //Move nowhere
            case NEW_UNKNOWN_HOST:
            case RESOLVING:
            case CONTACTING:
            case VALIDATING:
            case COPY_TIMEOUT_CHECK_HOST:
            case COPY_FAILED_CHECK_HOST:
            case PERM_QMASTER_SPOOL_DIR:
            case PERM_EXECD_SPOOL_DIR:
            case PERM_BDB_SPOOL_DIR:
            case BDB_SPOOL_DIR_EXISTS:
            case BDB_SPOOL_DIR_WRONG_FSTYPE:
            case ADMIN_USER_NOT_KNOWN:
            case PERM_JMX_KEYSTORE:
            case USED_QMASTER_PORT:
            case USED_EXECD_PORT:
            case USED_JMX_PORT:
            case READY_TO_INSTALL:
            case JVM_LIB_MISSING:
            case JVM_LIB_INVALID:
            case PROCESSING: break;

            //Success. Move to the success table
            case REACHABLE:
            case OK:
            case SUCCESS: {
                // At validation move nowhere
                if (!checkMode) {
                    targetTable = 1;
                }
                break;
            }

            // Failed! Move to failed table.
            case RESOLVABLE:
            case OPERATION_TIMEOUT:
            case UNKNOWN_ERROR:
            case CANCELED:
            case MISSING_FILE:
            case UNKNOWN_HOST:
            case UNREACHABLE:
            case COPY_TIMEOUT_INSTALL_COMPONENT:
            case COPY_FAILED_INSTALL_COMPONENT:
            case FAILED:
            case FAILED_ALREADY_INSTALLED_COMPONENT: //not used
            case FAILED_DEPENDENT_ON_PREVIOUS: {
                // At validation move nowhere
                if (!checkMode) {
                    targetTable = 2;
                }
                break;
            }

            //Unknown state
            default: throw new IllegalArgumentException("Unknown state: " + state);
        }

        /**
         * Move host to the proper panel
         */
        if (targetTable > -1) {
            ((HostTableModel)tables.get(targetTable).getModel()).addHost(host);

            if (installMode) {
                ((HostTableModel) tables.get(0).getModel()).removeHost(host);
            }
        }
    }

    /**
     * Sets the log of the given host
     * @param host The host
     * @param log The log to set
     */
    public void setHostLog(Host host, String log) {
        ((HostTableModel)tables.get(0).getModel()).setHostLog(host, log);
    }

    /**
     * Removes the given host
     * @param host The host to remove
     */
    public void removeHost(Host host) {
        for (JTable table : tables) {
            ((HostTableModel)table.getModel()).removeHost(host);
        }
    }

    /**
     * Adds the given hosts
     * @param hosts Hosts to add to the data
     */
    public void addHosts(List<Host> hosts) {
        resolveHosts(hosts);
    }

    /**
     * Install button's action. Main etry point of the installation procedure
     */
    private void installButtonActionPerformed() {

        clearTableSelections();

        // Check end fix variables
        fixVariables();

        //Installation must be started in a new Thread
        new Thread() {

            @Override
            public void run() {
                validateHostsAndInstall(true, false);
            }
        }.start();
    }

    /**
     * Validates the given host list then starts the installation
     * @param hosts The hosts have to be installed
     */
    private void validateHostsAndInstall(boolean install, boolean forced) {
        validateHostsAndInstall(lists.get(1), install, forced);
    }

    /**
     * Validates the given host list then starts the installation
     * @param hosts The hosts have to be installed
     */
    private void validateHostsAndInstall(HostList hosts, boolean install, boolean forced) {
        checkMode = true;
        //Disable the selecting host controls
        disableControls(true);
        enableBrowseButtons(false);

        setTablesEnabled(false);

        SwingUtilities.invokeLater(new Runnable() {

            public void run() {
                tabbedPane.setSelectedIndex(1);
            }
        });
       
        //Remove invalid components
        Host o;
        List<Host> tmpList = new ArrayList<Host>(); //need a copy of the hostlist
        List<Integer> indexList = new ArrayList<Integer>();
        for (int i = 0; i < hosts.size(); i++) {
            o = new Host(hosts.get(i));
            //Remove invalid components. They can't be installed!
            if (o.isShadowHost() && !isShadowdInst) {
                o.setShadowHost(false);
            }
            if (o.isExecutionHost() && !isExecdInst) {
                o.setExecutionHost(false);
            }
            if (o.isQmasterHost() && !isQmasterInst) {
                o.setQmasterHost(false);
            }
            if (o.isBdbHost() && !isBdbInst) {
                o.setBdbHost(false);
            }
            //Set new state
            o.setState(Host.State.READY_TO_INSTALL);
            if (o.hasAnyComponent()) {
                //tmpList.addUnchecked(o);
                tmpList.add(o);
                indexList.add(i);
            }
        }

        //And Check we have a components to install
        int numOfQmasterHost = 0;
        int numOfExecdHost = 0;
        int numOfShadowHost = 0;
        int numOfBdbHost = 0;
        int numOthers = 0;
        for (Host h : tmpList) {
            if (h.isQmasterHost()) {
                numOfQmasterHost++;
            }
            if (h.isExecutionHost()) {
                numOfExecdHost++;
            }
            if (h.isShadowHost()) {
                numOfShadowHost++;
            }
            if (h.isBdbHost()) {
                numOfBdbHost++;
            }
            if (!(h.isQmasterHost() && h.isShadowHost() && h.isBdbHost() &&  h.isExecutionHost()) && (h.isAdminHost() || h.isSubmitHost())) {
                numOthers++;
            }
        }
        if (!forced && numOfQmasterHost == 0 && numOfExecdHost == 0 && numOfShadowHost == 0 && numOfBdbHost == 0) {
            String message;
            int res=JOptionPane.YES_OPTION;
            if (numOthers > 0) {
                message = new VariableSubstitutor(idata.getVariables()).substituteMultiple(idata.langpack.getString("warning.only.admin.submit.to.install.message"), null);
                res=JOptionPane.showOptionDialog(this, message, getLabel("installer.warning"), JOptionPane.YES_NO_OPTION, JOptionPane.WARNING_MESSAGE, null,
                                                 new Object[]{getLabel("installer.yes"), getLabel("installer.no")}, getLabel("installer.no"));
            } else {
                message = new VariableSubstitutor(idata.getVariables()).substituteMultiple(idata.langpack.getString("warning.no.components.to.install.message"), null);
                JOptionPane.showOptionDialog(this, message, getLabel("installer.error"),
                    JOptionPane.CANCEL_OPTION, JOptionPane.ERROR_MESSAGE, null,
                    new Object[]{getLabel("installer.cancel")}, getLabel("installer.cancel"));
                res = JOptionPane.NO_OPTION;
            }

            //If Cancel or No, we don't proceed with the installation
            if (res == JOptionPane.NO_OPTION) {
                checkMode = false;
                disableControls(false);
                enableBrowseButtons(true);
                setTablesEnabled(true);
                return;
            }
        }

        // Compare host selection with the component selection if does not match warn user
        String componentString = "";
        String hostTypeString = "";

        if (isQmasterInst && numOfQmasterHost == 0) {
            componentString = getLabel("install.qmaster.label");
            hostTypeString = getLabel("column.qmaster.label");
        } else if (isExecdInst && numOfExecdHost == 0) {
            componentString = getLabel("install.execd.label");
            hostTypeString = getLabel("column.execd.label");
        } else if (isShadowdInst && numOfShadowHost == 0) {
            componentString = getLabel("install.shadowd.label");
            hostTypeString = getLabel("column.shadowd.label");
        } else if (isBdbInst && numOfBdbHost == 0) {
            componentString = getLabel("install.bdb.label");
            hostTypeString = getLabel("column.bdb.label");
        }

        if (!forced && !componentString.equals("")) {
            Properties props = new Properties();
            props.put("component", componentString);
            props.put("host.type", hostTypeString);

            VariableSubstitutor vs = new VariableSubstitutor(props);

            if (JOptionPane.NO_OPTION == JOptionPane.showOptionDialog(this, vs.substituteMultiple(getLabel("warning.comp.selected.but.no.host"), null),
                    getLabel("installer.warning"), JOptionPane.YES_NO_OPTION, JOptionPane.WARNING_MESSAGE, null,
                    new Object[]{getLabel("installer.yes"), getLabel("installer.no")}, getLabel("installer.no"))) {
               
                checkMode = false;
                disableControls(false);
                enableBrowseButtons(true);
                setTablesEnabled(true);

                return;
            }
            vs = null; // Don't use it any more
        }

        //Cancel host selection thread pool
        threadPool.shutdownNow();
        try {
            threadPool.awaitTermination(200, TimeUnit.MILLISECONDS);
        } catch (InterruptedException e) {
            Debug.error(e);
        }

        //Initialize new threadPool for the installation
        threadPool = (ThreadPoolExecutor) Executors.newFixedThreadPool(Util.INSTALL_THREAD_POOL_SIZE);
        threadPool.setThreadFactory(new TaskThreadFactory());       

        try {
            long started = 0;
            Host h;
            for (int i : indexList) {
                h = hosts.get(i);
                //Check only hosts that have real components
                if (h.isExecutionHost() || h.isShadowHost() || h.isQmasterHost() || h.isBdbHost()) {
                    try {
                       threadPool.execute(new ValidateHostTask(h, this, idata.getVariables()));
                       started++;
                    } catch (RejectedExecutionException e) {
                        setHostState(h, State.CANCELED);
                    }
                }
            }
            if (started > 0) {
                observer.setThreadPoolExecutors(threadPool);
                observer.observe();
            }

            while (threadPool.getCompletedTaskCount() < started && !threadPool.isTerminated()) {
                try {
                    Thread.sleep(500);
                } catch (InterruptedException ex) {
                }
            }

            int warningState = 0, noLocalSpoolWindows = 0;
            for (int i : indexList) {
                h = hosts.get(i);
                switch (h.getState()) {
                    case REACHABLE: break;
                    default: {
                        warningState++;
                    }
                }

                //If windows host, need to set a LOCAL_SPOOL_DIR if empty!
                if (!isExpressInst && h.getArchitecture().startsWith("win") &&
                        h.getSpoolDir().equals(idata.getVariable(VAR_EXECD_SPOOL_DIR))) {
                    noLocalSpoolWindows++;
                }
            }

            String msg = MessageFormat.format(getLabel("warning.invalid.hosts.message"), warningState, hosts.size());
            if (!forced && warningState > 0 && JOptionPane.NO_OPTION == JOptionPane.showOptionDialog(this, msg,
                    getLabel("installer.warning"), JOptionPane.YES_NO_OPTION, JOptionPane.WARNING_MESSAGE, null,
                    new Object[]{getLabel("installer.yes"), getLabel("installer.no")}, getLabel("installer.no"))) {

                disableControls(false);
                enableBrowseButtons(true);
                setTablesEnabled(true);

                return;
            }
           
            if (!forced && noLocalSpoolWindows > 0 && JOptionPane.NO_OPTION == JOptionPane.showOptionDialog(this, getLabel("warning.windows.needs.local.spooling.message"),
                    getLabel("installer.warning"), JOptionPane.YES_NO_OPTION, JOptionPane.WARNING_MESSAGE, null,
                    new Object[]{getLabel("installer.yes"), getLabel("installer.no")}, getLabel("installer.no"))) {

                disableControls(false);
                enableBrowseButtons(true);
                setTablesEnabled(true);

                return;
            }

        } catch (Exception e) {
            Debug.error(e);
        } finally {
            try {
                threadPool.awaitTermination(100, TimeUnit.MILLISECONDS);
                threadPool.purge();
            } catch (InterruptedException ex) {
            }

            checkMode = false;
        }

        if (install) {
            switchToInstallModeAndInstall(tmpList);
        }
    }

    private void switchToInstallModeAndInstall(final List<Host> hosts) {
        Debug.trace("INSTALL");

        //set install mode
        installMode = true;

        // Change panel heading
        if (parent != null) {
            parent.changeActualHeading(getLabel("HostPanel.installing.headline"));
        }

        // add.spooling.method
        if (idata.getVariable(VAR_SPOOLING_METHOD).equals("none")) {
            idata.setVariable(VAR_SPOOLING_METHOD, "berkeleydb");
        }

        validHostList = new ArrayList<Host>(hosts.size());
        validHostList.addAll(hosts);

        //Disable the selecting host controls
        disableControls(true);
        enableBrowseButtons(false);

        //Cancel host selection thread pool
        threadPool.shutdownNow();
        try {
            threadPool.awaitTermination(200, TimeUnit.MILLISECONDS);
        } catch (InterruptedException e) {
            Debug.error(e);
        }


        final List<String> adminHosts = new ArrayList<String>();
        final List<String> submitHosts = new ArrayList<String>();
        final List<String> allHosts = new ArrayList<String>();
        final List<String> allConnectUsers = new ArrayList<String>();
        final List<String> allShadowHosts = new ArrayList<String>();
        final List<String> allShadowConnectUsers = new ArrayList<String>();
        final List<String> possibleAdminHosts = new ArrayList<String>();
        //Remove hosts with no components + setup all admin/submit hosts
        HostList tmpList = new HostList();
        for (Host h : hosts) {
            //Doesn't add admin/submit hosts only
            if (h.getComponentString().length() > 0) {
                tmpList.add(h);
                if (!possibleAdminHosts.contains(h.getHostname())) {
                   possibleAdminHosts.add(h.getHostname());
                }
            }
            if (h.isAdminHost()) {
                adminHosts.add(h.getHostname());
                if (!possibleAdminHosts.contains(h.getHostname())) {
                   possibleAdminHosts.add(h.getHostname());
                }
            }
            if (h.isSubmitHost()) {
                submitHosts.add(h.getHostname());
            }
            if (h.isShadowHost()) {
                allShadowHosts.add(h.getHostname());
                allShadowConnectUsers.add(h.getConnectUser());
            }
            if (h.hasAnyComponent()) {
                allHosts.add(h.getHostname());
                allConnectUsers.add(h.getConnectUser());
            }
        }
        Host h, o;

        // Need qmaster host, try VAR_QMASTER_CONNECT_USER from the previous run first
        String qmasterConnectUser=(idata.getVariable(VAR_QMASTER_CONNECT_USER) != null) ? idata.getVariable(VAR_QMASTER_CONNECT_USER) : Util.DEF_CONNECT_USER;
        String QMASTER_HOST=Host.localHostName;
        try {
            QMASTER_HOST = Util.getQmasterHost(idata.getVariable(VAR_SGE_ROOT), idata.getVariable(VAR_SGE_CELL_NAME));
        } catch (FileNotFoundException ex) {
            if (isQmasterInst) {
                h = HostSelectionTableModel.getQmasterHost();
                if (h != null) {
                   QMASTER_HOST=h.getHostname();
                   qmasterConnectUser=h.getConnectUser();
                }
            }
        } catch (IOException ex) {
            //Unexpected
            Debug.error(ex);
        }

        firstTaskHost = new Host(Host.Type.HOSTNAME, Host.localHostName, Host.HOST_TYPE_ALL, true, false);
        firstTaskHost.setIp("all");
        firstTaskHost.setArchitecture("all");
        firstTaskHost.setConnectUser(qmasterConnectUser);

        //Create install list
        final HostList installList = new HostList();
        boolean haveFirstTask = false;
        //Sort the hosts so that BDB is first and qmaster second
        //Find bdb and put it to the beggining
        if (isBdbInst) {
            for (int i = 0; i < tmpList.size(); i++) {
                h = tmpList.get(i);
                if (h.isBdbHost()) {
                    o = new Host(h);
                    boolean hasMore = false;
                    //Clear selection
                    if (o.isQmasterHost()) {
                        o.setQmasterHost(false);
                        hasMore = true;
                    }
                    if (o.isExecutionHost()) {
                        o.setExecutionHost(false);
                        hasMore = true;
                    }
                    if (o.isShadowHost()) {
                        o.setShadowHost(false);
                        hasMore = true;
                    }
                    installList.addUnchecked(new Host(o));
                    if (hasMore) {
                        o.setAdminHost(false);
                        o.setSubmitHost(false);
                    }
                    tmpList.remove(o);
                    break;
                }
            }
        }

        if (isQmasterInst) {
            //Find qmaster and put it as next
            for (int i = 0; i < tmpList.size(); i++) {
                h = tmpList.get(i);
                if (h.isQmasterHost()) {
                    qmasterConnectUser=h.getConnectUser();
                    o = new Host(h);
                    boolean hasMore = false;
                    //Clear selection
                    if (o.isBdbHost()) {
                        o.setBdbHost(false);
                        hasMore = true;
                    }
                    if (o.isExecutionHost()) {
                        o.setExecutionHost(false);
                        hasMore = true;
                    }
                    if (o.isShadowHost()) {
                        o.setShadowHost(false);
                        hasMore = true;
                    }
                    installList.addUnchecked(new Host(o));
                    if (hasMore) {
                        o.setAdminHost(false);
                        o.setSubmitHost(false);
                    }
                    tmpList.remove(o);
                    //We add additional prerequisite task (first task)
                    installList.addUnchecked(firstTaskHost);
                    haveFirstTask = true;
                    break;
                }
            }
        }
        //TODO: Sort the rest of the list alphabetically

        //Add first task if none added so far
        if (!haveFirstTask) {
            //We add additional prerequisite task (first task)
            //After issue 2973 fixed in 6.2u3 needed also for shadowds
            installList.addUnchecked(firstTaskHost);
        }

        //We cannot install shadowds in parallel!
        if (isShadowdInst) {
            int size = 0;
            int i=0;
            boolean listSizeChanged;
            //Find shadowds and put them as next
            while (i < tmpList.size()) {
                listSizeChanged = false;
                h = tmpList.get(i);
                if (h.isShadowHost()) {
                    o = new Host(h);
                    boolean hasMore = false;
                    //Clear selection
                    if (o.isBdbHost()) {
                        o.setBdbHost(false);
                        hasMore = true;
                    }
                    if (o.isExecutionHost()) {
                        o.setExecutionHost(false);
                        hasMore = true;
                    }
                    if (o.isQmasterHost()) {
                        o.setQmasterHost(false);
                        hasMore = true;
                    }
                    installList.addUnchecked(new Host(o));
                    if (hasMore) {
                        o.setAdminHost(false);
                        o.setSubmitHost(false);
                    }
                    size = tmpList.size();
                    tmpList.remove(o);
                    if (tmpList.size() != size) {
                        listSizeChanged = true;
                    }
                }
                //Increment i only if we didn't remove a host completely
                if (!listSizeChanged) {
                   i++;
                }
            }
        }
       
        //Copy the rest (execds)
        for (Host host : tmpList) {
            installList.addUnchecked(new Host(host));
        }
        //Add the final task
        lastTaskHost = new Host(Host.Type.HOSTNAME, QMASTER_HOST, Host.HOST_TYPE_ALL, false, true);
        lastTaskHost.setIp("all");
        lastTaskHost.setArchitecture("all");
        lastTaskHost.setConnectUser(qmasterConnectUser);
        installList.addUnchecked(lastTaskHost);

        //Standalone bdb, qmaster installations do not need the first task (for others we might need to copy CSP certs)
        if (installList.size() == 3) {
            Host firstHost = installList.get(0);
            if (firstHost.isBdbHost() || firstHost.isQmasterHost()) {
                installList.removeUnchecked(firstTaskHost);
            } else {
                firstTaskHost.setDisplayName(firstHost.getHostname());
                firstTaskHost.setIp(firstHost.getIp());
                firstTaskHost.setArchitecture(firstHost.getArchitecture());
            }
            lastTaskHost = installList.get(installList.size() - 1);
            lastTaskHost.setDisplayName(firstHost.getHostname());
            lastTaskHost.setIp(firstHost.getIp());
            lastTaskHost.setArchitecture(firstHost.getArchitecture());
        }

        /**
         * Generate global auto_conf files depending on the different local execd spool dirs
         */
        // Seperate execution hosts depending on their execd spool dir
        Hashtable<String, ArrayList<Host>> hostSelection = new Hashtable<String, ArrayList<Host>>();
        String localExecdSpoolDir = "";
        for (Host host : installList) {
            if (host.isExecutionHost()) {
                localExecdSpoolDir = host.getSpoolDir();

                if (!hostSelection.containsKey(localExecdSpoolDir)) {
                    ArrayList<Host> hostList = new ArrayList<Host>();
                    hostList.add(host);
                    hostSelection.put(localExecdSpoolDir, hostList);
                } else {
                    hostSelection.get(localExecdSpoolDir).add(host);
                }
            }
        }

        // Construct the file names
        VariableSubstitutor vs = new VariableSubstitutor(idata.getVariables());

        String autoConfTempFile = vs.substituteMultiple(idata.getVariable(VAR_AUTO_CONF_TEMP_FILE), null);
        String autoConfFile = vs.substituteMultiple(idata.getVariable(VAR_AUTO_CONF_FILE), null);

        autoConfFile = autoConfFile + "_" + Util.generateTimeStamp();

        //Set global values
        idata.setVariable(VAR_QMASTER_CONNECT_USER, qmasterConnectUser);
        idata.setVariable(VAR_GUI_SHELL_OPTIONS, CmdExec.getShellOptions(idata.getVariable(VAR_SHELL_NAME)));
        // Generate the auto_conf file for every different local execd spool dirs
        int index = 0;
        String outputFilePostfix = "";
        boolean firstRun = true; // Let a run even if there was no execution host selected
        for (Enumeration<String> enumer = hostSelection.keys(); firstRun || enumer.hasMoreElements();) {
            firstRun = false;

            if (enumer.hasMoreElements()) {
                localExecdSpoolDir = enumer.nextElement();
                idata.setVariable(VAR_EXEC_HOST_LIST, Util.getHostNames(hostSelection.get(localExecdSpoolDir), " "));
                idata.setVariable(VAR_EXEC_HOST_LIST_RM, Util.getHostNames(hostSelection.get(localExecdSpoolDir), " "));
            }

            // put the
            if (!enumer.hasMoreElements()) {
                idata.setVariable(VAR_SHADOW_HOST_LIST, Util.getHostNames(Util.getHosts(installList, Util.SgeComponents.shadow), " "));
                idata.setVariable(VAR_ADMIN_HOST_LIST, Util.getHostNames(new ArrayList<Host>(), adminHosts, " "));
                idata.setVariable(VAR_SUBMIT_HOST_LIST, Util.getHostNames(new ArrayList<Host>(), submitHosts, " "));

                outputFilePostfix = "";
            } else {
                idata.setVariable(VAR_SHADOW_HOST_LIST, "");
                idata.setVariable(VAR_ADMIN_HOST_LIST, "");
                idata.setVariable(VAR_SUBMIT_HOST_LIST, "");

                outputFilePostfix = "_" + String.valueOf(index);
                index++;
            }

            if (localExecdSpoolDir.equals(idata.getVariable(VAR_EXECD_SPOOL_DIR))) {
                idata.setVariable(VAR_EXECD_SPOOL_DIR_LOCAL, "");
            } else {
                idata.setVariable(VAR_EXECD_SPOOL_DIR_LOCAL, localExecdSpoolDir);
            }

            outputFilePostfix = autoConfFile + outputFilePostfix + ".conf";
            try {
                Util.fillUpTemplate(autoConfTempFile, outputFilePostfix, idata.getVariables());
                Debug.trace("Generating auto_conf file: '" + outputFilePostfix + "'.");
            } catch (Exception ex) {
                Debug.error("Failed to generate auto_conf file: '" + outputFilePostfix + "'." + ex);
            }
        }

        /**
         * Hide the input panel, there is no way back anyway!
         */
        SwingUtilities.invokeLater(new Runnable() {

            public void run() {
                hostRB.setVisible(false);
                hostTF.setVisible(false);
                addB.setVisible(false);
                fileB.setVisible(false);
                fileRB.setVisible(false);
                componentSelectionPanel.setVisible(false);
                statusBar.setVisible(false);
            }
        });
       

        /**
         * Build install table
         */

        // Initialize column tooltips
        String[] installHeaders = getInstallLabelVars();
        String[] headerTooltips = new String[installHeaders.length];
        for (int i = 0; i < installHeaders.length; i++) {
            headerTooltips[i] = getTooltip(installHeaders[i]);
        }

        //TODO: Do not overwrite old values tabbedpane, lists, hosts, etc..
        //Create a new tabbed pane for installation
        tabbedPane.invalidate();
        tabbedPane.removeAll();
        lists = new Vector<HostList>();
        tables = new Vector<HostTable>();
        HostList list;
        HostTable table;
        for (int i = 0; i < INSTALL_TABS.length; i++) {
            list = (i == 0) ? installList : new HostList();
            table = new HostTable(this);
            lists.add(list);
            tables.add(table);
            table.setModel(new HostInstallTableModel(list, getInstallHeaders(), getInstallClassTypes()));

            table.setTableHeader(new TooltipTableHeader(table.getColumnModel(), headerTooltips));
            JTableHeader header = table.getTableHeader();
            final SortedColumnHeaderRenderer headerRenderer = new SortedColumnHeaderRenderer(
                    header,
                    getImageIcon("columns.sorted.asc"),
                    getImageIcon("columns.sorted.desc"));
            for (int col = 0; col < table.getColumnCount(); col++) {
                table.getColumnModel().getColumn(col).setHeaderRenderer(headerRenderer);
            }

            table.getColumn(getLabel("column.progress.label")).setCellRenderer(new StateCellRenderer());
            table.getColumn(getLabel("column.log.label")).setCellRenderer(new LogButtonCellRenderer(getLabel("cell.log.label"), getLabel("cell.nolog.label")));
            table.getColumn(getLabel("column.log.label")).setCellEditor(new LogButtonCellEditor(getLabel("cell.log.label")));

            // 6786350 - gui-installer: when moving columns the sorting stays in the column index not on the column ifself
            table.getColumnModel().addColumnModelListener(new TableColumnModelListener() {
                public void columnAdded(TableColumnModelEvent e) {}
                public void columnRemoved(TableColumnModelEvent e) {}
                public void columnMoved(TableColumnModelEvent e) {
                    headerRenderer.setPressedColumn(e.getToIndex());
                    headerRenderer.setSelectedColumn(e.getToIndex(), headerRenderer.getSelectedColumnState());
                }
                public void columnMarginChanged(ChangeEvent e) {}
                public void columnSelectionChanged(ListSelectionEvent e) {}
            });

            header.addMouseListener(new TableHeaderListener(header, headerRenderer));

            table.getModel().addTableModelListener(new TableModelListener() {

                public void tableChanged(TableModelEvent e) {
                    HostPanel.this.tableChanged();
                }
            });

            tabbedPane.addTab(INSTALL_TABS[i] + " (" + lists.get(i).size() + ")", new JScrollPane(tables.get(i)));
            tabbedPane.setToolTipTextAt(i, INSTALL_TABS_TOOLTIPS[i]);
        }
        tabbedPane.validate();

        //Installation must be started in a new Thread
        new Thread() {

            @Override
            public void run() {
                startInstallation(installList, possibleAdminHosts, adminHosts, submitHosts, allHosts, allConnectUsers, allShadowHosts, allShadowConnectUsers);
            }
        }.start();
    }

    private void startInstallation(HostList installList, final List<String> possibleAdminHosts, final List<String> adminHosts, final List<String> submitHosts, final List<String> allHosts, final List<String> allConnectUsers, final List<String> allShadowHosts, final List<String> allShadowConnectUsers) {
        //Initialize new threadPool for the installation
        threadPool = (ThreadPoolExecutor) Executors.newFixedThreadPool(Util.INSTALL_THREAD_POOL_SIZE);
        threadPool.setThreadFactory(new TaskThreadFactory());
        //We need a new executor for shadowdTasks (only 1 task at single moment)
        singleThreadPool = (ThreadPoolExecutor) Executors.newFixedThreadPool(1);
        singleThreadPool.setThreadFactory(new TaskThreadFactory());

        observer.setThreadPoolExecutors(new ThreadPoolExecutor[]{threadPool, singleThreadPool});
        observer.setTaskCount(installList.size());
        observer.observe();

        //Create a copy of the install list for the last_task
        HostList initialInstallList = new HostList();
        for (Host h: installList) {
            initialInstallList.addUnchecked(new Host(h));
        }

        boolean wait = false;
        long completed = 0, started = 0;
        try {
            for (Host h : installList) {
                //Skip the last task is done in finally
                if (h.isLastTask()) {
                    continue;
                }
                Properties variablesCopy = new Properties();
                variablesCopy.putAll(idata.getVariables());

                h.setComponentVariables(variablesCopy);
                if (h.isBdbHost() || h.isQmasterHost()) {
                    wait = true;
                    completed = singleThreadPool.getCompletedTaskCount();
                }

                variablesCopy.setProperty(VAR_JVM_LIB_PATH, h.getJvmLibPath());
                variablesCopy.setProperty(VAR_ADDITIONAL_JVM_ARGS, h.getJvmAddArgs());

                //All windows need windows specified varibles to be set as well
                if (h.getArchitecture().startsWith("win")) {
                    variablesCopy.setProperty(VAR_WINDOWS_SUPPORT, "true");
                    variablesCopy.setProperty(VAR_WIN_DOMAIN_ACCESS, "true");
                    //TODO: What about WIN_ADMIN_USER?
                }

                //BDB, Qmaster, Shadowd installation go to special singlethreadPool
                if (h.isBdbHost() || h.isQmasterHost() || h.isShadowHost()) {
                    try {
                        if (h.isQmasterHost() && h.getInstallTimeout() < 360000) {
                            //Qmaster installation can wait up to 5mins to qmaster to startup
                            h.setInstallTimeout(Util.DEF_INSTALL_TIMEOUT + 300000);
                        }
                        singleThreadPool.execute(new InstallTask(h, this, variablesCopy, localizedMessages));
                    } catch (RejectedExecutionException e) {
                        setHostState(h, State.CANCELED);
                    }
                } else if (h.isFirstTask()) {
                    //This is a first execd/shadowd task and there were no other BDB or qmaster components
                    //Need to create a first task that will add all hosts as admin hosts
                    Properties vars = new Properties();
                    vars.putAll(idata.getVariables());
                    vars.put(VAR_ALL_ADMIN_HOSTS, Util.listToString(possibleAdminHosts));
                    vars.setProperty(VAR_ALL_CSPHOSTS, "");
                    vars.setProperty(VAR_ALL_COPYUSERS, "");
                    //Only for CSP mode
                    if (vars.getProperty("add.product.mode").equalsIgnoreCase("csp")) {
                        List<String> cspList = allHosts;
                        String allHostsButCurrent = "";
                        String allUsers = "";
                        // Construct a reverse odered list so the local host will be the last one
                        for (int i = cspList.size() - 1; i >= 0; i--) {
                                allHostsButCurrent += cspList.get(i).trim() + " ";
                                allUsers += allConnectUsers.get(i).trim() + " ";
                            }
                        vars.setProperty(VAR_ALL_CSPHOSTS, allHostsButCurrent.trim());
                        vars.setProperty(VAR_ALL_COPYUSERS, allUsers.trim());
                    } else if (vars.getProperty(VAR_SGE_JMX).equalsIgnoreCase("true") && vars.getProperty(VAR_JMX_SSL).equalsIgnoreCase("true")) {
                        //Copy certs to all shadow hosts (need the keystores to be able to start JMX thread)
                        String allHostsButCurrent = "";
                        String allUsers = "";
                        // Construct a reverse odered list so the local host will be the last one
                        for (int i = allShadowHosts.size() - 1; i >= 0; i--) {
                                allHostsButCurrent += allShadowHosts.get(i).trim() + " ";
                                allUsers += allShadowConnectUsers.get(i).trim() + " ";
                            }
                        vars.setProperty(VAR_ALL_CSPHOSTS, allHostsButCurrent.trim());
                        vars.setProperty(VAR_ALL_COPYUSERS, allUsers.trim());
                    }
                    vars.put(VAR_FIRST_TASK, "true");
                    wait = true;

                    try {
                        //And execute the first task in the singleThreadPool
                        singleThreadPool.execute(new InstallTask(h, this, vars, localizedMessages));
                    } catch (RejectedExecutionException e) {
                        setHostState(h, State.CANCELED);
                    }
                } else {
                    try {
                        //Only execd get installed in parallel
                        threadPool.execute(new InstallTask(h, this, variablesCopy, localizedMessages));
                    } catch (RejectedExecutionException e) {
                        setHostState(h, State.CANCELED);
                    }
                }

                started++;
                //In case the task is a BDB or qmaster host, we have to wait for sucessful finish!
                while (wait) {
                    if (singleThreadPool.getCompletedTaskCount() >= completed + 1 && !singleThreadPool.isTerminated()) {
                        wait = false;
                        //If bdb, qmaster, prereq tasks fail => it's over!
                        if (h.getState() != Host.State.SUCCESS) {
                            for (Host host : installList) {
                                setHostLog(host, "FAILED: " + MessageFormat.format(localizedMessages.getProperty("msg.previous.dependent.install.failed"), h.getComponentString()));
                                setHostState(host, State.FAILED_DEPENDENT_ON_PREVIOUS);
                            }
                            observer.setTaskCount(-1);
                            return;
                        }
                        completed++;
                    } else if (singleThreadPool.isTerminated()){
                        //There is not guarantee that the currently executing task will be interrupted. It may also finish as SUCCESS or FAILED!
                        for (Host host : installList) {
                            setHostLog(host, "CANCELED: " + MessageFormat.format(localizedMessages.getProperty("msg.install.canceled"), ""));
                            setHostState(host, State.CANCELED);
                        }
                        observer.setTaskCount(-1);
                        return;
                    } else {
                        try {
                            Thread.sleep(200);
                        } catch (InterruptedException ex) {
                        }
                    }
                }
            }
        } catch (Exception e) {
            Debug.error(e);
        } finally {
            //Wait until all tasks have finished
            while (threadPool.getCompletedTaskCount() + singleThreadPool.getCompletedTaskCount() < started && !(threadPool.isTerminated() && singleThreadPool.isTerminated())) {
                try {
                    Thread.sleep(100);
                } catch (InterruptedException ex) {
                }
            }
            //Execute final task to setup correct submit and remove invalid admin hosts silently on local host
            Properties vars = new Properties();
            if (lastTaskHost == null) {
                Debug.error("lastTaskHost not set!");
                throw new IllegalArgumentException("lastTaskHost not set!");
            } else if (lastTaskHost.getState() != Host.State.FAILED_DEPENDENT_ON_PREVIOUS) {
                //Run the last task only if we have the lastTaskHost and we are not in FAILED_DEPENDENT_ON_PREVIOUS state
                vars.putAll(idata.getVariables());
                vars.put(VAR_ALL_ADMIN_HOSTS, Util.listToString(adminHosts));
                vars.put(VAR_ALL_HOSTS, Util.listToString(allHosts));
                vars.put(VAR_ALL_SUBMIT_HOSTS, Util.listToString(submitHosts));
                vars.put(VAR_LAST_TASK, "true");
                //And execute the last task in the singleThreadPool
                try {
                    singleThreadPool.execute(new InstallTask(lastTaskHost, this, vars, localizedMessages));
                } catch (RejectedExecutionException e) {
                    setHostState(lastTaskHost, State.CANCELED);
                }
                started++;
                //Wait until it's finished
                while (threadPool.getCompletedTaskCount() + singleThreadPool.getCompletedTaskCount() < started && !singleThreadPool.isTerminated()) {
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException ex) {
                    }
                }
            }
            //If we have failed hosts we go to the Failed tab
            if (((HostInstallTableModel) tables.get(2).getModel()).getRowCount() > 0) {
                tabbedPane.setSelectedIndex(2);
            } else { //Go to succeeded tab
                tabbedPane.setSelectedIndex(1);
            }

            // Create report for the ResultPanel
            createReport();

            //Let's cleanup
            observer.setTaskCount(-1);
           
            try {
                threadPool.awaitTermination(100, TimeUnit.MILLISECONDS);
                singleThreadPool.awaitTermination(100, TimeUnit.MILLISECONDS);
                threadPool.purge();
                singleThreadPool.purge();
            } catch (InterruptedException ex) {
            }
            threadPool.shutdown();
            singleThreadPool.shutdown();
        }
    }

    // Variables declaration - do not modify//GEN-BEGIN:variables
    private javax.swing.JButton addB;
    private javax.swing.JCheckBox adminCB;
    private javax.swing.JScrollPane allHostsScrollPane;
    private javax.swing.JCheckBox bdbCB;
    private javax.swing.JButton cancelB;
    private javax.swing.JPanel componentSelectionPanel;
    private javax.swing.JCheckBox execCB;
    private javax.swing.JButton fileB;
    private javax.swing.JRadioButton fileRB;
    private javax.swing.ButtonGroup hostButtonGroup;
    private javax.swing.JRadioButton hostRB;
    private javax.swing.JTextField hostTF;
    private javax.swing.JProgressBar progressBar;
    private javax.swing.JCheckBox qmasterCB;
    private javax.swing.JCheckBox shadowCB;
    private javax.swing.JLabel statusBar;
    private javax.swing.JCheckBox submitCB;
    private javax.swing.JTabbedPane tabbedPane;
    // End of variables declaration//GEN-END:variables
    private JTextField lastSelectedTF;

    /**
     * Makes xml output from the panel's data for the automated instalaltion
     */
    @Override
    public void makeXMLData(XMLElement panelRoot) {
        Map<String, String> entryMap = new HashMap<String, String>();

        for (Host h : validHostList) {
            entryMap.put(h.getHostname(), h.toStringInstance());
        }

        new HostPanelAutomationHelper(entryMap).makeXMLData(idata, panelRoot);
    }
}
TOP

Related Classes of com.sun.grid.installer.gui.HostPanel

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.