/******************************************************************************* * * Copyright (c) 2004-2009 Oracle Corporation. * * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * * Kohsuke Kawaguchi * * *******************************************************************************/ package hudson.util; import hudson.CloseProofOutputStream; import hudson.console.ConsoleNote; import hudson.console.HudsonExceptionNote; import hudson.model.TaskListener; import hudson.remoting.RemoteOutputStream; import org.kohsuke.stapler.framework.io.WriterOutputStream; import java.io.Closeable; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.OutputStream; import java.io.OutputStreamWriter; import java.io.PrintStream; import java.io.PrintWriter; import java.io.Serializable; import java.io.UnsupportedEncodingException; import java.io.Writer; import java.nio.charset.Charset; import java.util.logging.Level; import java.util.logging.Logger; /** * {@link TaskListener} that generates output into a single stream. * * <p> This object is remotable. * * @author Kohsuke Kawaguchi */ public class StreamTaskListener extends AbstractTaskListener implements Serializable, Closeable { private PrintStream out; private Charset charset; /** * @deprecated as of 1.349 The caller should use * {@link #StreamTaskListener(OutputStream, Charset)} to pass in the charset * and output stream separately, so that this class can handle encoding * correctly, or use {@link #fromStdout()} or {@link #fromStderr()}. */ public StreamTaskListener(PrintStream out) { this(out, null); } public StreamTaskListener(OutputStream out) { this(out, null); } public StreamTaskListener(OutputStream out, Charset charset) { try { if (charset == null) { this.out = (out instanceof PrintStream) ? (PrintStream) out : new PrintStream(out, false); } else { this.out = new PrintStream(out, false, charset.name()); } this.charset = charset; } catch (UnsupportedEncodingException e) { // it's not very pretty to do this, but otherwise we'd have to touch too many call sites. throw new Error(e); } } public StreamTaskListener(File out) throws IOException { this(out, null); } public StreamTaskListener(File out, Charset charset) throws IOException { // don't do buffering so that what's written to the listener // gets reflected to the file immediately, which can then be // served to the browser immediately this(new FileOutputStream(out), charset); } public StreamTaskListener(Writer w) throws IOException { this(new WriterOutputStream(w)); } /** * @deprecated as of 1.349 Use {@link #NULL} */ public StreamTaskListener() throws IOException { this(new NullStream()); } public static StreamTaskListener fromStdout() { return new StreamTaskListener(System.out, Charset.defaultCharset()); } public static StreamTaskListener fromStderr() { return new StreamTaskListener(System.err, Charset.defaultCharset()); } public PrintStream getLogger() { return out; } private PrintWriter _error(String prefix, String msg) { out.print(prefix); out.println(msg); // the idiom in Hudson is to use the returned writer for writing stack trace, // so put the marker here to indicate an exception. if the stack trace isn't actually written, // HudsonExceptionNote.annotate recovers gracefully. try { annotate(new HudsonExceptionNote()); } catch (IOException e) { // for signature compatibility, we have to swallow this error } return new PrintWriter( charset != null ? new OutputStreamWriter(out, charset) : new OutputStreamWriter(out), true); } public PrintWriter error(String msg) { return _error("ERROR: ", msg); } public PrintWriter error(String format, Object... args) { return error(String.format(format, args)); } public PrintWriter fatalError(String msg) { return _error("FATAL: ", msg); } public PrintWriter fatalError(String format, Object... args) { return fatalError(String.format(format, args)); } public void annotate(ConsoleNote ann) throws IOException { ann.encodeTo(out); } private void writeObject(ObjectOutputStream out) throws IOException { out.writeObject(new RemoteOutputStream(new CloseProofOutputStream(this.out))); out.writeObject(charset == null ? null : charset.name()); } private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { out = new PrintStream((OutputStream) in.readObject(), true); String name = (String) in.readObject(); charset = name == null ? null : Charset.forName(name); } public void close() throws IOException { out.close(); } /** * Closes this listener and swallows any exceptions, if raised. * * @since 1.349 */ public void closeQuietly() { try { close(); } catch (IOException e) { LOGGER.log(Level.WARNING, "Failed to close", e); } } private static final long serialVersionUID = 1L; private static final Logger LOGGER = Logger.getLogger(StreamTaskListener.class.getName()); }