Logo Search packages:      
Sourcecode: henplus version File versions  Download package

SpoolCommand.java

/*
 * This is free software, licensed under the Gnu Public License (GPL)
 * get a copy from <http://www.gnu.org/licenses/gpl.html>
 * $Id: SpoolCommand.java,v 1.4 2004/02/01 18:46:40 hzeller Exp $ 
 * author: Henner Zeller <H.Zeller@acm.org>
 */
package henplus.commands;

import henplus.SQLSession;
import henplus.HenPlus;
import henplus.AbstractCommand;
import henplus.OutputDevice;
import henplus.PrintStreamOutputDevice;

import java.util.Stack;
import java.util.Date;
import java.io.PrintStream;
import java.io.OutputStream;
import java.io.FileOutputStream;
import java.io.IOException;

/**
 * prepared ..
 */
00025 public final class SpoolCommand extends AbstractCommand {
    private final Stack/*<OutputDevice>*/ _outStack;
    private final Stack/*<OutputDevice>*/ _msgStack;

    /**
     * returns the command-strings this command can handle.
     */
00032     public String[] getCommandList() {
      return new String[] {
          "spool"
      };
    }
    
    public SpoolCommand(HenPlus hp) {
        _outStack  = new Stack/*<OutputDevice>*/();
        _msgStack = new Stack/*<OutputDevice>*/();
        _outStack.push(hp.getOutputDevice());
        _msgStack.push(hp.getMessageDevice());
    }

00045     public boolean requiresValidSession(String cmd) { return false; }

    /**
     * execute the command given.
     */
00050     public int execute(SQLSession currentSession, String cmd, String param) {
        param = param.trim();
        try {
            if ("off".equals(param.toLowerCase())) {
                closeSpool();
            }
            else if (param.length() > 0) {
                openSpool(param);
            }
            else {
                return SYNTAX_ERROR;
            }
        }
        catch (Exception e) {
            e.printStackTrace();
            return EXEC_FAILED;
        }
      return SUCCESS;
    }

    /**
     * combine the current output from the stack with the given output,
     * use this as current output and return it.
     */
00074     private OutputDevice openStackedDevice(Stack/*<OutputDevice>*/ stack,
                                           OutputDevice newOut) {
        OutputDevice origOut = (OutputDevice) stack.peek();
        OutputDevice outDevice = new StackedDevice(origOut, newOut);
        stack.push(outDevice);
        return outDevice;
    }

    /**
     * close the top device on the stack and return the previous.
     */
00085     private OutputDevice closeStackedDevice(Stack/*<OutputDevice>*/ stack) {
        OutputDevice out = (OutputDevice) stack.pop();
        out.close();
        return (OutputDevice) stack.peek();
    }

    private void openSpool(String filename) throws IOException {
        // open file
        OutputDevice spool = new PrintStreamOutputDevice(new PrintStream(new FileOutputStream(filename)));
        HenPlus.getInstance().setOutput(openStackedDevice(_outStack, spool),
                                        openStackedDevice(_msgStack, spool));
        HenPlus.msg().println("-- open spool at " + new Date());
    }

    private boolean closeSpool() throws IOException {
        if (_outStack.size() == 1) {
            HenPlus.msg().println("no open spool.");
            return false;
        }
        HenPlus.msg().println("-- close spool at " + new Date());
        HenPlus.getInstance().setOutput(closeStackedDevice(_outStack),
                                        closeStackedDevice(_msgStack));
        return true;
    }

00110     public String getShortDescription() {
        return "log output to a file";
    }

00114     public String getSynopsis(String cmd) {
        return "spool <filename>|off";
    }

00118     public String getLongDescription(String cmd) { 
      String dsc;
      dsc= "\tIf command is followed by a filename, opens a file\n"
            +"\tand writes all subsequent output not only to the terminal\n"
            +"\tbut as well to this file. With\n"
            +"\t spool off\n"
            +"\tspooling is stopped and the file is closed. The spool\n"
            +"\tcommand works recursivly, i.e. you can open more than one \n"
            +"\tfile, and you have to close each of them with 'spool off'\n";
      return dsc;
    }

    /**
     * A stream that writes to two output streams. On close, only
     * the second, stacked stream is closed.
     */
00134     private static class StackedDevice implements OutputDevice {
        private final OutputDevice _a;
        private final OutputDevice _b;

        public StackedDevice(OutputDevice a, OutputDevice b) {
            _a = a;
            _b = b;
        }

        public void flush() {
            _a.flush(); 
            _b.flush();
        }
        public void write(byte[] buffer, int off, int len) {
            _a.write(buffer, off, len);
            _b.write(buffer, off, len);
        }
        public void print(String s) {
            _a.print(s);
            _b.print(s);
        }
        public void println(String s) {
            _a.println(s);
            _b.println(s);
        }
        public void println() {
            _a.println();
            _b.println();
        }
        
        public void attributeBold() {
            _a.attributeBold();
            _b.attributeBold();
        }
        public void attributeGrey() {
            _a.attributeGrey();
            _b.attributeGrey();
        }
        public void attributeReset() {
            _a.attributeReset();
            _b.attributeReset();
        }
        
        /** closes _only_ the (stacked) second stream */
00178         public void close() {
            _b.close();
        }
    }
}

/*
 * Local variables:
 * c-basic-offset: 4
 * compile-command: "ant -emacs -find build.xml"
 * End:
 */

Generated by  Doxygen 1.6.0   Back to index