Package com.sun.grid.jgdi.util.shell

Source Code of com.sun.grid.jgdi.util.shell.QConfCommand

/*___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.jgdi.util.shell;

import com.sun.grid.jgdi.configuration.EventClient;
import com.sun.grid.jgdi.JGDIException;
import com.sun.grid.jgdi.configuration.ClusterQueue;
import com.sun.grid.jgdi.configuration.ComplexEntry;
import com.sun.grid.jgdi.configuration.ComplexEntryImpl;
import com.sun.grid.jgdi.configuration.ExecHost;
import com.sun.grid.jgdi.configuration.GEObject;
import com.sun.grid.jgdi.configuration.Hostgroup;
import com.sun.grid.jgdi.configuration.JGDIAnswer;
import com.sun.grid.jgdi.configuration.ResourceQuotaSet;
import com.sun.grid.jgdi.configuration.ShareTree;
import com.sun.grid.jgdi.configuration.ShareTreeImpl;
import com.sun.grid.jgdi.configuration.UserSet;
import com.sun.grid.jgdi.configuration.UserSetImpl;
import com.sun.grid.jgdi.util.OutputTable;
import com.sun.grid.jgdi.util.shell.editor.EditorParser;
import com.sun.grid.jgdi.util.shell.editor.EditorUtil;
import com.sun.grid.jgdi.util.shell.editor.GEObjectEditor;
import java.beans.IntrospectionException;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import static com.sun.grid.jgdi.util.JGDIShell.getResourceString;
import static com.sun.grid.jgdi.util.shell.OptionAnnotation.MAX_ARG_VALUE;

/**
*
*/
@CommandAnnotation(value = "qconf")
public class QConfCommand extends QConfCommandGenerated {

    public void run(String[] args) throws Exception {
        if (args.length == 0) {
            throw new IllegalArgumentException("Invalid number of arguments");
        }
        parseAndInvokeOptions(args);
    }

    //-help
    @OptionAnnotation(value = "-help", min = 0)
    public void printUsage(final OptionInfo oi) throws JGDIException {
        out.println(getUsage());
    }

    //SUBMITHOST
    @OptionAnnotation(value = "-as", extra = MAX_ARG_VALUE)
    public void addSubmitHost(final OptionInfo oi) throws JGDIException {
        if (!isHostReachable(oi)) {
            oi.optionDone();
            return;
        }
        final String hostName = oi.getFirstArg();
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        jgdi.addSubmitHostWithAnswer(hostName, answers);
        printAnswers(answers);
    }

    @OptionAnnotation(value = "-ds", extra = MAX_ARG_VALUE)
    public void deleteSubmitHost(final OptionInfo oi) throws JGDIException {
        if (!isHostReachable(oi)) {
            oi.optionDone();
            return;
        }
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        int size = oi.getArgs().size();
        final String[] hosts = oi.getArgs().toArray(new String[size]);
        oi.optionDone();
        jgdi.deleteSubmitHostsWithAnswer(hosts, answers);
        printAnswers(answers);
    }

    //ADMINHOST
    @OptionAnnotation(value = "-ah", extra = MAX_ARG_VALUE)
    public void addAdminHost(final OptionInfo oi) throws JGDIException {
        if (!isHostReachable(oi)) {
            oi.optionDone();
            return;
        }
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        String hostName = oi.getFirstArg();
        jgdi.addAdminHostWithAnswer(hostName, answers);
        printAnswers(answers);
    }

    @OptionAnnotation(value = "-dh", extra = MAX_ARG_VALUE)
    public void deleteAdminHost(final OptionInfo oi) throws JGDIException {
        if (!isHostReachable(oi)) {
            oi.optionDone();
            return;
        }
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        int size = oi.getArgs().size();
        final String[] hosts = oi.getArgs().toArray(new String[size]);
        oi.optionDone();
        jgdi.deleteAdminHostsWithAnswer(hosts, answers);
        printAnswers(answers);
    }

    //SHARETREE
    @OptionAnnotation(value = "-astree", min = 0)
    public void addShareTree(final OptionInfo oi) throws JGDIException {
        ShareTree stree = null;
        try {
            stree = jgdi.getShareTree("Root");
        } catch (JGDIException je) {
            // ignore no sharetree defined and fill in a template element instead
        }
        if (stree == null) {
            stree = new ShareTreeImpl("template");
        }
        String treeAsText = showShareTree(stree, "Root", true);
        String userTypedText = runEditor(treeAsText);
        if (userTypedText != null) {
            List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
            ShareTreeImpl node = new ShareTreeImpl();
            node = GEObjectEditor.updateObjectWithText(jgdi, node, userTypedText);
            // pw.println(node.dump());
            jgdi.updateShareTreeWithAnswer(node, answers);
            printAnswers(answers);
        }
        oi.optionDone();
    }

    @OptionAnnotation(value = "-Astree")
    public void addShareTreeFromFile(final OptionInfo oi) throws JGDIException {
        String inputText = readFile(oi);
        out.println("NOT IMPLEMENTED"); //TODO LP: Implement
    }

    // -mstree same as -astree
    @OptionAnnotation(value = "-mstree", min = 0)
    public void modifyShareTree(final OptionInfo oi) throws JGDIException {
        addShareTree(oi);
    }

    // -Mstree same as -Astree
    @OptionAnnotation(value = "-Mstree")
    public void modifyShareTreeFromFile(final OptionInfo oi) throws JGDIException {
        addShareTreeFromFile(oi);
    }

    @OptionAnnotation(value = "-sstree", min = 0)
    public void showShareTree(final OptionInfo oi) throws JGDIException {
        ShareTree stree = jgdi.getShareTree("Root");
        out.println(showShareTree(stree, "Root", true));
    }

    @OptionAnnotation(value = "-dstree", min = 0)
    public void deleteShareTree(final OptionInfo oi) throws JGDIException {
        ShareTree empty = new ShareTreeImpl(true);
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        jgdi.deleteShareTreeWithAnswer(answers);
        printAnswers(answers);
    }

    /**
     * Show sharetree node
     * @param oi
     * @return
     * @throws com.sun.grid.jgdi.JGDIException
     */
    String showShareTreeNode(final OptionInfo oi) throws JGDIException {
        final String name = oi.getFirstArg();
        ShareTree stree = jgdi.getShareTree(name);
        return showShareTree(stree, name, false);
    }

    /*
     * Show sharetree node
     */
    private static String showShareTree(ShareTree node, final String name, final boolean showTree) throws JGDIException {
        StringBuilder sb = new StringBuilder();
        List<ShareTree> sharetree = new LinkedList<ShareTree>();
        List<ShareTree> childList;
        String childStr;
        String stName;
        ShareTree tempTree;
        sharetree.add(node);
        while (!sharetree.isEmpty()) {
            node = sharetree.remove(0);
            //Add children to queue
            childList = new LinkedList<ShareTree>();
            childList.addAll(node.getChildrenList());
            //Sort the list by ID
            Collections.sort(childList, new Comparator<ShareTree>() {

                public int compare(ShareTree a, ShareTree b) {
                    int na = a.getId();
                    int nb = b.getId();
                    return (na >= nb) ? ((na == nb) ? 0 : 1) : -1;
                }
            });
            childStr = "";
            for (ShareTree tree : childList) {
                sharetree.add(tree);
                childStr += tree.getId() + ",";
            }
            //For show node
            if (!showTree) {
                stName = node.getName();
                stName = stName.equals("Root") ? "" : stName;
                sb.append("/" + stName + "=" + node.getShares() + "\n");
                //For show tree
            } else {
                sb.append("id=" + node.getId() + "\n");
                sb.append("name=" + node.getName() + "\n");
                sb.append("type=" + node.getType() + "\n");
                sb.append("shares=" + node.getShares() + "\n");
                childStr = (childStr.length() > 0) ? childStr.substring(0, childStr.length() - 1) : "NONE";
                sb.append("childnodes=" + childStr + "\n");
            }
        }
        return sb.toString();
    }

    //-tsm
    @OptionAnnotation(value = "-tsm", min = 0)
    public void triggerSchedulerMonitoring(OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        jgdi.triggerSchedulerMonitoringWithAnswer(answers);
        printAnswers(answers);
    }

    //-clearusage
    @OptionAnnotation(value = "-clearusage", min = 0)
    public void clearUsage(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        jgdi.clearShareTreeUsageWithAnswer(answers);
        //TODO LP: Bug - got no answers
        printAnswers(answers);
    }

    //-cq
    @OptionAnnotation(value = "-cq", extra = MAX_ARG_VALUE)
    public void cleanQueue(final OptionInfo oi) throws JGDIException {
        final List<String> args = oi.getArgs();
        oi.optionDone();
        final String[] queues = args.toArray(new String[oi.getArgs().size()]);
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        jgdi.cleanQueuesWithAnswer(queues, answers);
        printAnswers(answers);
    }

    //-kec
    @OptionAnnotation(value = "-kec", extra = MAX_ARG_VALUE)
    public void killEventClient(final OptionInfo oi) throws JGDIException {
        final List<String> args = oi.getArgs();
        oi.optionDone();
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        int[] ids = new int[args.size()];
        int i = 0;
        for (String arg : args) {
            if (arg.equals("all")) {
                jgdi.killAllEventClientsWithAnswer(answers);
                printAnswers(answers);
                return;
            } else {
                try {
                    ids[i] = Integer.parseInt(arg);
                } catch (NumberFormatException nfe) {
                    throw new IllegalArgumentException(arg + " is not a valid event client id");
                }
            }
            i++;
        }
        jgdi.killEventClientsWithAnswer(ids, answers);
        printAnswers(answers);
    }

    //-km
    @OptionAnnotation(value = "-km", min = 0)
    public void killMaster(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        jgdi.killMasterWithAnswer(answers);
        printAnswers(answers);
    }

    //-ks
    @OptionAnnotation(value = "-ks", min = 0)
    public void killScheduler(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        jgdi.killSchedulerWithAnswer(answers);
        printAnswers(answers);
    }

    //-sds
    @OptionAnnotation(value = "-sds", min = 0)
    public void showDetachedSettings(final OptionInfo oi) throws JGDIException {
        String sds;
        sds = jgdi.showDetachedSettingsAll();
        //TODO LP: Check correct behaviour, for now expect that sds!="" is an error
        if (sds != null && sds.length() > 0) {
            out.print(sds);
            setExitCode(1);
        }
        oi.optionDone();
    }

    //-secl
    @OptionAnnotation(value = "-secl", min = 0)
    public void showEventClientList(final OptionInfo oi) throws JGDIException {
        List<EventClient> evcl = jgdi.getEventClientList();
        if (!evcl.isEmpty()) {
            OutputTable table = new OutputTable(com.sun.grid.jgdi.configuration.EventClient.class);
            try {
                table.addCol("id", "ID", 8, OutputTable.Column.RIGHT);
                table.addCol("name", "NAME", 15, OutputTable.Column.LEFT);
                table.addCol("host", "HOST", 24, OutputTable.Column.LEFT);
            } catch (IntrospectionException ex) {
                ex.printStackTrace();
            }
            table.printHeader(out);
            out.write("--------------------------------------------------\n");
            for (EventClient evc : evcl) {
                table.printRow(out, (Object) evc);
            }
        } else {
            err.println(getErrorMessage("NoObjectFound", oi.getOd().getOption()));
            setExitCode(getCustomExitCode("NoObjectFound", oi.getOd().getOption()));
        }
    }

    //-sep
    @OptionAnnotation(value = "-sep", min = 0)
    @SuppressWarnings(value = "unchecked")
    public void showProcessors(final OptionInfo oi) throws JGDIException {
        List<ExecHost> hosts;
        hosts = jgdi.getExecHostList();
        String name;
        String arch;
        int cpu;
        int totalCpu = 0;
        out.println("HOST                      PROCESSOR        ARCH");
        out.println("===============================================");
        //TODO LP: How about we sort the hosts by name first?
        for (ExecHost host : hosts) {
            name = host.getName();
            if (name.equals("global") || name.equals("template")) {
                continue;
            }
            cpu = host.getProcessors();
            totalCpu += cpu;
            if (host.isSetLoad()) {
                arch = host.getLoad("arch");
            } else {
                arch = "";
            }
            out.printf("%-25.25s %9d %11s%n", name, cpu, arch);
        }
        out.println("===============================================");
        out.printf("%-25.25s %9d%n", "SUM", totalCpu);
    }

    //-sss
    @OptionAnnotation(value = "-sss", min = 0)
    public void showSchedulerState(final OptionInfo oi) throws JGDIException {
        out.println(jgdi.getSchedulerHost());
    }

    //-ke
    @OptionAnnotation(value = "-ke", extra = MAX_ARG_VALUE)
    public void killExecd(final OptionInfo oi) throws JGDIException {
        final List<String> args = oi.getArgs();
        oi.optionDone();
        killExecd(false, args);
    }

    //-kej
    @OptionAnnotation(value = "-kej", extra = MAX_ARG_VALUE)
    public void killExecdWithJobs(final OptionInfo oi) throws JGDIException {
        final List<String> args = oi.getArgs();
        oi.optionDone();
        killExecd(true, args);
    }

    private void killExecd(final boolean terminateJobs, final List<String> args) throws JGDIException {
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        for (String host : args) {
            if (host.equals("all")) {
                jgdi.killAllExecdsWithAnswer(terminateJobs, answers);
                printAnswers(answers);
                return;
            }
        }
        jgdi.killExecdWithAnswer(args.toArray(new String[args.size()]), terminateJobs, answers);
        printAnswers(answers);
    }

    //HOSTGROUP
    //-shgrp_tree
    @OptionAnnotation(value = "-shgrp_tree")
    public void showHostgroupTree(final OptionInfo oi) throws JGDIException {
        final List<String> args = oi.getArgs();
        oi.optionDone();
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        int i = 0;
        while (i < args.size()) {
            //Prepare the hgroup
            String hgroup = args.get(i++);
            int level = 0;
            Hostgroup obj = null;
            try {
                obj = jgdi.getHostgroupWithAnswer(hgroup, answer);
                printAnswers(answer);
            } catch (JGDIException ex) {
                err.println(ex.getMessage()); //TODO LP: Check if the message is a correct one, what about exitCode?
                setExitCode(ex.getExitCode());
            }
            //Print the tree
            if (obj != null) {
                showHostgroupTree(obj, "", "   ");
            } else {
                err.println(getErrorMessage("InvalidObjectArgument", oi.getOd().getOption(), hgroup));
                setExitCode(getCustomExitCode("InvalidObjectArgument", oi.getOd().getOption()));
            }
        }
    }

    //-shgrp_resolved
    @OptionAnnotation(value = "-shgrp_resolved")
    @SuppressWarnings(value = "unchecked")
    public void showHostgroupResolved(final OptionInfo oi) throws JGDIException {
        final List<String> args = oi.getArgs();
        oi.optionDone();
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        int i = 0;
        while (i < args.size()) {
            //Prepare the hgroup
            String hgroup = args.get(i++);
            int level = 0;
            Hostgroup obj = null;
            try {
                obj = jgdi.getHostgroupWithAnswer(hgroup, answer);
                printAnswers(answer);
            } catch (JGDIException ex) {
                err.println(ex.getMessage()); //TODO LP: Check if the message is a correct one
            }
            //Print the tree
            if (obj != null) {
                List<String> hnames = new LinkedList<String>();
                printHostgroupResolved(hnames, obj);
                String res = "";
                for (String hname : hnames) {
                    res += hname + " ";
                }
                out.println(res.substring(0, res.length() - 1));
            } else {
                err.println(getErrorMessage("InvalidObjectArgument", oi.getOd().getOption(), hgroup));
                setExitCode(getCustomExitCode("InvalidObjectArgument", oi.getOd().getOption()));
            }
        }
    }

    //TODO LP: Remove recursion in shgrp_tree
    private void showHostgroupTree(Hostgroup obj, String prefix, final String tab) {
        out.println(prefix + obj.getName());
        prefix += tab;

        for (String hgroup : (List<String>) obj.getHostList()) {
            //Another hroup
            if (hgroup.startsWith("@")) {
                try {
                    obj = jgdi.getHostgroup(hgroup);
                } catch (JGDIException ex) {
                    err.println(ex.getMessage());
                    setExitCode(ex.getExitCode());
                }
                showHostgroupTree(obj, prefix, tab);
            } else {
                out.println(prefix + hgroup);
            }
        }
    }

    //TODO: Clients use postorder, better to use sort?
    private void printHostgroupResolved(List<String> result, Hostgroup obj) {
        LinkedList<Hostgroup> queue = new LinkedList<Hostgroup>();
        queue.add(obj);
        while (!queue.isEmpty()) {
            obj = queue.remove(0);
            for (String hgroup : obj.getHostList()) {
                //Another hroup
                if (hgroup.startsWith("@")) {
                    try {
                        obj = jgdi.getHostgroup(hgroup);
                        queue.add(obj);
                    } catch (JGDIException ex) {
                        err.println(ex.getMessage());
                        setExitCode(ex.getExitCode());
                    }
                } else {
                    if (!result.contains(hgroup)) {
                        result.add(hgroup);
                    }
                }
            }
        }
        //Collections.sort(result);
    }

    //COMPLEXENTRY
    //-mc
    @OptionAnnotation(value = "-mc", min = 0)
    public void modifyComplexEntry(final OptionInfo oi) throws JGDIException {
        String text = runEditor(showComplexes());
        modifyComplexes(text);
    }

    //-Mc
    @OptionAnnotation(value = "-Mc")
    public void modifyComplexEntryFromFile(final OptionInfo oi) throws JGDIException {
        String inputText = readFile(oi);
        modifyComplexes(inputText);
    }

    //-sc
    @OptionAnnotation(value = "-sc", min = 0)
    public void showComplexEntry(final OptionInfo oi) throws JGDIException {
        out.print(showComplexes());
    }

    /**
     * Updates the complex entry list based on the text
     * @param text
     */
    @SuppressWarnings(value = "unchecked")
    private void modifyComplexes(final String text) throws JGDIException {
        //Now parse lines and fields ignore whitespaces join lines on \
        List<List<String>> lineList = EditorParser.tokenizeToList(text);
        Map<String, ComplexEntry> newCEMap = new HashMap<String, ComplexEntry>();
        String elem;
        int val;
        for (List<String> singleLine : lineList) {
            elem = singleLine.get(0);

            //We check if the line starts with # and skip it
            if (elem.startsWith("#")) {
                continue;
            }
            if (singleLine.size() != 8) {
                throw new IllegalArgumentException("Expected 8 elements: name, shortcut, type, relop, requestable, consumable, default, urgency");
            }
            ComplexEntry ce = new ComplexEntryImpl(true);
            ce.setName(elem);
            ce.setShortcut(singleLine.get(1));

            val = ce.typeToInt(singleLine.get(2));
            //TODO LP: deny invalid
            ce.setValtype(val);

            val = ce.opToInt(singleLine.get(3));
            //TODO LP: deny invalid
            ce.setRelop(val);

            val = ce.reqToInt(singleLine.get(4));
            //TODO LP: deny invalid
            ce.setRequestable(val);

            ce.setConsumable(Util.getYesNoJobAsInt(singleLine.get(5)));

            ce.setDefault(singleLine.get(6));

            //val = Integer.parseInt((String) singleLine.get(7));
            ce.setUrgencyWeight(singleLine.get(7));

            newCEMap.put(elem, ce);
        }
        //TODO LP: Better to have jgdi.replaceCEList(List newList) and make the checks on qmaster side
        List<ComplexEntry> toModify = new LinkedList<ComplexEntry>();
        List<ComplexEntry> toDelete = new LinkedList<ComplexEntry>();
        List<ComplexEntry> origList = jgdi.getComplexEntryList();

        String name;
        String shortCut;
        for (ComplexEntry ce : origList) {
            name = ce.getName();
            //Check if existing value is modified
            if (newCEMap.containsKey(name)) {
                if (!ce.equalsCompletely(newCEMap.get(name))) {
                    toModify.add(newCEMap.get(name));
                }
                newCEMap.remove(name);
            } else {
                toDelete.add(ce);
            }
        }
        //Add new complexes
        for (String key : newCEMap.keySet()) {
            ComplexEntry ce = newCEMap.get(key);
            try {
                jgdi.addComplexEntry(ce);
            } catch (JGDIException ex) {
                err.println(ex.getMessage());
                setExitCode(ex.getExitCode());
            }
        }
        //Modify existing
        for (ComplexEntry ce : toModify) {
            try {
                jgdi.updateComplexEntry(ce);
            } catch (JGDIException ex) {
                err.println(ex.getMessage());
                setExitCode(ex.getExitCode());
            }
        }
        //Remove not defined anymore
        for (ComplexEntry ce : toDelete) {
            try {
                jgdi.deleteComplexEntry(ce);
            } catch (JGDIException ex) {
                err.println(ex.getMessage());
                setExitCode(ex.getExitCode());
            }
        }
    }

    private String showComplexes() {
        StringBuilder sb = new StringBuilder();
        List<ComplexEntry> cList = null;
        try {
            cList = jgdi.getComplexEntryList();
        } catch (JGDIException ex) {
            return ex.getMessage();
        }
        //Format is:nameLen+3 shortcutLen+3 typeLen+3 relopLen+3 requestableLen+8 consumableLen+7 defaultLen+3 urgencyLen+4
        int nameLen = 0;
        int shortcutLen = 0;
        int typeLen = 0;
        int relopLen = 2;
        int requestableLen = 0;
        int consumableLen = 0;
        int defaultLen = 0;
        int urgencyLen = 0;
        //Sort the list alphabetically
        List<ComplexEntry> complexList = sortListByName(cList);
        ComplexEntry ceDummy = new ComplexEntryImpl(true);
        //Need to first get the maximum column lengths
        for (ComplexEntry complex : complexList) {
            nameLen = Math.max(nameLen, complex.getName().length());
            shortcutLen = Math.max(shortcutLen, complex.getShortcut().length());
            typeLen = Math.max(typeLen, ceDummy.typeToString(complex.getValtype()).length());
            requestableLen = Math.max(requestableLen, (complex.getRequestable() == 1) ? 2 : 3); //length of YES, NO
            consumableLen = Math.max(consumableLen, (complex.getConsumable() != 0) ? 3 : 2); //length of YES/JOB, NO);
            defaultLen = Math.max(defaultLen, complex.getDefault().length());
            urgencyLen = Math.max(urgencyLen, complex.getUrgencyWeight().length());
        }
        //Now format the columns
        String header0 = String.format("%-19.19s %-10.10s %-11.11s %-5.5s %-11.11s %-10.10s %-8.8s %-8s", "#name", "shortcut", "type", "relop", "requestable", "consumable", "default", "urgency");
        StringBuilder header1 = new StringBuilder("#");
        for (int j = 0; j < header0.length() - 1; j++) {
            header1.append("-");
        }
        sb.append(header0 + "\n");
        sb.append(header1 + "\n");
        String val;
        //And finally print the columns
        for (ComplexEntry complex : complexList) {
            val = String.format("%-19.19s %-10.10s %-11.11s %-5.5s %-11.11s %-10.10s %-8.8s %-8s", complex.getName(),
                    complex.getShortcut(), ceDummy.typeToString(complex.getValtype()), ceDummy.opToString(complex.getRelop()),
                    ceDummy.reqToString(complex.getRequestable()), Util.getYesNoJobAsString(complex.getConsumable()), complex.getDefault(),
                    complex.getUrgencyWeight());
            sb.append(val + "\n");
        }
        sb.append("# >#< starts a comment but comments are not saved across edits --------\n");
        return sb.toString();
    }
   
    private static List sortListByName(final List<? extends GEObject> list) {
        Collections.sort(list, new Comparator<GEObject>() {

            public int compare(GEObject o1, GEObject o2) {
                if (o1 == null && o2 == null) {
                    return 0;
                }
                if (o1 == null) {
                    return -1;
                }
                if (o2 == null) {
                    return 1;
                }
                return o1.getName().compareTo(o2.getName());
            }
        });
        return list;
    }

    //MANAGER
    @OptionAnnotation(value = "-am", extra = MAX_ARG_VALUE)
    public void addManager(final OptionInfo oi) throws JGDIException {
        final String name = oi.getFirstArg();
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        jgdi.addManagerWithAnswer(name, answer);
        printAnswers(answer);
    }

    @OptionAnnotation(value = "-dm", extra = MAX_ARG_VALUE)
    public void deleteManager(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        String[] a = new String[0];
        int size = oi.getArgs().size();
        final String[] names = oi.getArgs().toArray(new String[size]);
        oi.optionDone();
        jgdi.deleteManagersWithAnswer(names, answers);
        printAnswers(answers);
    }

    //OPERATOR
    @OptionAnnotation(value = "-ao", extra = MAX_ARG_VALUE)
    public void addOperator(final OptionInfo oi) throws JGDIException {
        final String name = oi.getFirstArg();
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        jgdi.addOperatorWithAnswer(name, answer);
        printAnswers(answer);
    }

    @OptionAnnotation(value = "-do", extra = MAX_ARG_VALUE)
    public void deleteOperator(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answers = new LinkedList<JGDIAnswer>();
        String[] a = new String[0];
        int size = oi.getArgs().size();
        final String[] names = oi.getArgs().toArray(new String[size]);
        oi.optionDone();
        jgdi.deleteOperatorsWithAnswer(names, answers);
        printAnswers(answers);
    }

    //RESOURCE QUOTA SET
    /**
     *   Implements qconf -mrqs option
     *   @param  oi <b>OptionInfo</b> option enviroment object
     *   @throws JGDIException on any error on the GDI layer
     */
    @OptionAnnotation(value = "-mrqs", min = 0, extra = MAX_ARG_VALUE)
    public void modifyResourceQuotaSet(final OptionInfo oi) throws JGDIException {
        //TODO LP: Needs spceial editor handling, displays more objects at once
        ResourceQuotaSet obj;
        List<ResourceQuotaSet> rqsList = new LinkedList<ResourceQuotaSet>();
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        String textToEdit = "";

        oi.optionDone();
        if (oi.getArgs().size() == 0) {
            rqsList = jgdi.getResourceQuotaSetListWithAnswer(answer);
            printAnswers(answer);
        } else {
            for (String arg : oi.getArgs()) {
                obj = jgdi.getResourceQuotaSetWithAnswer(arg, answer);
                rqsList.add(obj);
                printAnswers(answer);
                answer.clear();
            }
        }
        //Create text to be displayed in the editor
        for (ResourceQuotaSet rqs : rqsList) {
            textToEdit += GEObjectEditor.getConfigurablePropertiesAsText(rqs);
        }

        String userTypedText = runEditor(textToEdit);
        if (userTypedText != null) {
            //TODO LP: Handle the multiple objects. Need special parser for the userTypedText
            //GEObjectEditor.updateObjectWithText(jgdi, obj, userTypedText);
            //jgdi.updateResourceQuotaSetWithAnswer(obj, answer);
            //printAnswers(answer);
            throw new IllegalStateException("NOT IMPLEMENTED");
        }
        oi.optionDone();
    }

    /**
     *   Implements qconf -srqs option
     *   @param  oi <b>OptionInfo</b> option enviroment object
     *   @throws JGDIException on any error on the GDI layer
     */
    @OptionAnnotation(value = "-srqs", min = 0, extra = MAX_ARG_VALUE)
    public void showResourceQuotaSet(final OptionInfo oi) throws JGDIException {
        ResourceQuotaSet obj;
        List<ResourceQuotaSet> rqsList = new LinkedList<ResourceQuotaSet>();
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        String text = "";

        oi.optionDone();
        if (oi.getArgs().size() == 0) {
            rqsList = jgdi.getResourceQuotaSetListWithAnswer(answer);
            oi.optionDone();
            printAnswers(answer);
        } else {
            for (String arg : oi.getArgs()) {
                obj = jgdi.getResourceQuotaSetWithAnswer(arg, answer);
                rqsList.add(obj);
                printAnswers(answer);
                answer.clear();
            }
        }
        //Show correct error message if list is empty
        if (rqsList.size() == 0) {
            err.println(getErrorMessage("NoObjectFound", oi.getOd().getOption()));
            setExitCode(getCustomExitCode("NoObjectFound", oi.getOd().getOption()));
            return;
        }
        //Create text to be displayed in the editor
        for (ResourceQuotaSet rqs : rqsList) {
            text += GEObjectEditor.getConfigurablePropertiesAsText(rqs);
        }
        out.print(text);
    }

    //USERSET
    //-au
    @OptionAnnotation(value = "-au", min = 2, extra = MAX_ARG_VALUE)
    public void addUserSet(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        final List<String> setList = oi.getLastOriginalArgList();
        final List<String> userList = oi.getArgs();
        UserSet obj;
        //This is the only call we want to make
        oi.optionDone();
        if (userList.size() == 0) {
            //We are missing the set list and used usernames as sets
            String msg = getErrorMessage("LessArguments", oi.getOd().getOption(), new ArrayList(setList));
            throw new JGDIException(msg, getCustomExitCode("LessArguments", oi.getOd().getOption()));
        }
        boolean isNew;
        //We add all the users to all the usersets
        for (String set : setList) {
            obj = jgdi.getUserSet(set);
            isNew = false;
            if (obj == null) {
                obj = new UserSetImpl(true);
                obj.setName(set);
                isNew = true;
            }
            for (String user: userList) {
                //TODO LP: CR XXXXXX addEntries curently add already existing elements
                //Missing messages added user to set
                obj.addEntries(user);
            }
            if (isNew) {
                jgdi.addUserSetWithAnswer(obj, answer);
            } else {
                jgdi.updateUserSetWithAnswer(obj, answer);
            }
            printAnswers(answer);
            answer.clear();
        }
    }

    //-du
    @OptionAnnotation(value = "-du", min = 2, extra = MAX_ARG_VALUE)
    public void deleteUserSet(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        final List<String> setList = oi.getLastOriginalArgList();
        final List<String> userList = oi.getArgs();
        UserSet obj;
       
        //This is the only call we want to make
        oi.optionDone();
        if (userList.size() == 0) {
            //We are missing the set list and used usernames as sets
            String msg = getErrorMessage("LessArguments", oi.getOd().getOption(), new ArrayList(setList));
            throw new JGDIException(msg, getCustomExitCode("LessArguments", oi.getOd().getOption()));
        }
        //We delete form all the sets
        for (String set : setList) {
            obj = jgdi.getUserSetWithAnswer(set, answer);
            printAnswers(answer);
            answer.clear();
            if (obj == null) {
                err.println(getErrorMessage("InvalidObjectArgument", oi.getOd().getOption(), set));
                setExitCode(getCustomExitCode("InvalidObjectArgument", oi.getOd().getOption()));
                //TODO LP: Missing message about invalid name
                continue;
            }
            for (String user: userList) {
                //TODO LP: CR XXXXXX addEntries curently add already existing elements
                //Also messages are coming from the client!
                //user "user3" is not in access list "set1"
                //deleted user "user3" from access list "set2"
                obj.removeEntries(user);
            }
            jgdi.updateUserSetWithAnswer(obj, answer);
            printAnswers(answer);
            answer.clear();
        }
    }

    //-dul
    @OptionAnnotation(value = "-dul", min = 1, extra = MAX_ARG_VALUE)
    public void deleteUserSetList(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        final String setName = oi.getFirstArg();
        UserSet obj = jgdi.getUserSetWithAnswer(setName, answer);
        printAnswers(answer);
        if (obj == null) {
            err.println(getErrorMessage("InvalidObjectArgument", oi.getOd().getOption(), setName));
            setExitCode(getCustomExitCode("InvalidObjectArgument", oi.getOd().getOption()));
            return;
        }
        answer.clear();
        jgdi.deleteUserSetWithAnswer(obj, answer);
        printAnswers(answer);
    }
   
    //EXECHOST
    @OptionAnnotation(value = "-ae", min = 0, extra = 1)
    public void addExecHost(final OptionInfo oi) throws JGDIException {
        if (isHostReachable(oi)) {
            oi.optionDone();
            return;
        }
        String arg = oi.getFirstArg();
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        //TODO LP: Remove this ugly(inconsistent) behaviour in client and JGDI (then can be generated as well)
        ExecHost obj = jgdi.getExecHostWithAnswer(arg, answer);
        //TODO LP: Qmaster check candidate
        printAnswers(answer);
        answer.clear();
        if (obj == null) {
            err.println(getErrorMessage("InvalidObjectArgument", oi.getOd().getOption() , arg));
            setExitCode(getCustomExitCode("InvalidObjectArgument", oi.getOd().getOption()));
            return;
        }
        String userTypedText = runEditor(GEObjectEditor.getConfigurablePropertiesAsText(obj));
        if (userTypedText != null) {
            GEObjectEditor.updateObjectWithText(jgdi, obj, userTypedText);
            jgdi.addExecHostWithAnswer(obj, answer);
            printAnswers(answer);
        }
        //oi.optionDone();
    }
   
    //ATTR options
    @OptionAnnotation(value = "-dattr", min = 4, extra = 0)
    public void deleteAttribute(final OptionInfo oi) throws JGDIException {
        List<JGDIAnswer> answer = new LinkedList<JGDIAnswer>();
        String name = "";
       
        String objectName = oi.getFirstArg().toLowerCase();
        String attrName = oi.getFirstArg().toLowerCase();
        String attrValue = oi.getFirstArg();
        String objectIdList = oi.getFirstArg();
       
        if (objectName.equals("queue")) {
            ClusterQueue q = jgdi.getClusterQueueWithAnswer(objectIdList, answer);
            printAnswers(answer);
            answer.clear();
            //TODO LP: This exception should have been thrown from the jgdi.getClusterQueueWithAnswer for invalid queue name!
            if (q == null) {
                throw new JGDIException("denied: cluster queue \""+objectIdList+"\" does not exist",1);
            }
            //TODO LP: Investigate if we should be allowed to add/modify/replace/delete ANY atribute!!!
            String val = GEObjectEditor.getPropertyAsText(q, EditorUtil.PROPERTIES_ALL, attrName);
            int pos = val.indexOf(attrValue);
            if (pos == -1) {
                throw new JGDIException("JGDI Error: value \""+attrValue+"\" is not present in attribute \""+attrName+"\" for "+objectName+" object "+"\""+objectIdList+"\"",1);
            }
            name = val.split(" ")[0].trim();
            val = val.replaceFirst(attrValue, "");
            //Look if we removed last value and add NONE in necessary
            if (val.trim().substring(name.length()).length() == 0) {
                val = name + " NONE";
            }
            //Change the object ifself
            q = GEObjectEditor.updateObjectWithText(jgdi, q, val);
            //And send it back to qmaster
            jgdi.updateClusterQueueWithAnswer(q, answer);
            printAnswers(answer);
        } else {
            throw new JGDIException("JGDI Error: Unsupported object_name \""+objectName+"\"",1);
        }
    }
   
   
    //OVERRIDES - special handling to have same behaviour as in clients
    //TODO: Fix the clients remove this special handling

    //CALENDAR
    @OptionAnnotation(value = "-acal", min = 0, extra = 1)
    @Override
    public void addCalendar(final OptionInfo oi) throws JGDIException {
        //Handling client inconsistancy. Does not open editor when no arg given
        if (hasNoArgument(oi)) {
            return;
        }
        //otherwise handle adding object
        super.addCalendar(oi);
    }

    //CHECKPOINT
    @OptionAnnotation(value = "-ackpt", min = 0, extra = 1)
    @Override
    public void addCheckpoint(final OptionInfo oi) throws JGDIException {
        //Handling client inconsistancy. Does not open editor when no arg given
        if (hasNoArgument(oi)) {
            return;
        }
        //otherwise handle adding of the object
        super.addCheckpoint(oi);
    }
   
    //CLUSTERQUEUE
    //TODO: Template is not a know queue, but 'qconf -sq' prints the template in clients
    @OptionAnnotation(value = "-sq", defaultStringArg="template", min = 0, extra = MAX_ARG_VALUE)
    @Override
    public void showClusterQueue(OptionInfo oi) throws JGDIException {
        super.showClusterQueue(oi);
    }
   
    //CONFIGURATION
    @OptionAnnotation(value = "-aconf", min = 0, extra = 1)
    @Override
    public void addConfiguration(final OptionInfo oi) throws JGDIException {
        if (isHostReachable(oi)) {
            oi.optionDone();
            return;
        }
        //otherwise handle adding of the object
        super.addConfiguration(oi);
    }

    @OptionAnnotation(value = "-sconf", defaultStringArg="global", min = 0, extra = MAX_ARG_VALUE)
    @Override
    public void showConfiguration(OptionInfo oi) throws JGDIException {
        super.showConfiguration(oi);
    }

    //PARALLEL ENVIRONMENT
    @OptionAnnotation(value = "-sp", min = 1, extra = 0)
    @Override
    public void showParallelEnvironment(final OptionInfo oi) throws JGDIException {
        super.showParallelEnvironment(oi);
    }

    //PROJECT
    @OptionAnnotation(value = "-sprj", min = 1, extra = 0)
    @Override
    public void showProject(final OptionInfo oi) throws JGDIException {
        super.showProject(oi);
    }

    //HOSTGROUP -  additional check if the argument starts with @ character
    @OptionAnnotation(value = "-ahgrp", min = 0, extra = 1)
    @Override
    public void addHostgroup(final OptionInfo oi) throws JGDIException {
        if (!hasValidHostgroupName(oi)) {
            return;
        }
        super.addHostgroup(oi);
    }
   
    //EXECHOST    
    @OptionAnnotation(value = "-me", min = 1, extra = 0)
    @Override
    public void modifyExecHost(final OptionInfo oi) throws JGDIException {
        if (!isHostReachable(oi)) {
            oi.optionDone();
            return;
        }
        super.modifyExecHost(oi);
    }
   
   
   
    /**
     * Helper method. Checks if option has an argument(s)
     * return hasNoArgument
     */
    private boolean hasNoArgument(OptionInfo oi) {
        String option = oi.getOd().getOption();
        if (oi.getArgs().size() == 0) {
            err.println(getErrorMessage("NoArgument", option));
            setExitCode(getCustomExitCode("NoArgument", option));
            return true;
        }
        return false;
    }

    /**
     * Helper method. Checks if option's next argument is a valid hostgroup name (starting with @).
     * return hasNoArgument
     */
    private boolean hasValidHostgroupName(OptionInfo oi) {
        List<String> args = oi.getArgs();
        OptionDescriptor od = oi.getOd();
        if (args != null && args.size() > 0) {
            String hgrp = args.get(0);
            if (!hgrp.startsWith("@")) {
                String msg = getErrorMessage("InvalidObjectArgument", od.getOption(), hgrp);
                err.println(msg);
                setExitCode(getCustomExitCode("InvalidObjectArgument", od.getOption()));
                return false;
            }
        }
        return true;
    }

    /**
     * Helper method. Checks if option's next argument is a reachable host
     * return isHostReachable
     */
    private boolean isHostReachable(OptionInfo oi) {
        List<String> args = oi.getArgs();
        OptionDescriptor od = oi.getOd();
        if (args != null && args.size() > 0) {
            String host = args.get(0);
            try {
                if (InetAddress.getByName(host) != null) {
                    return true;
                }
            } catch (UnknownHostException ex) {
                String msg = getErrorMessage("UnreachableHost", od.getOption(), host);
                err.println(msg);
                setExitCode(getCustomExitCode("UnreachableHost", od.getOption()));
                return false;
            }
        }
        return false;
    }
}
TOP

Related Classes of com.sun.grid.jgdi.util.shell.QConfCommand

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.