DemuxOutputStream.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.io.output;

  18. import java.io.IOException;
  19. import java.io.OutputStream;

  20. import org.apache.commons.io.IOUtils;

  21. /**
  22.  * Forwards data to a stream that has been associated with this thread.
  23.  */
  24. public class DemuxOutputStream extends OutputStream {

  25.     private final InheritableThreadLocal<OutputStream> outputStreamThreadLocal = new InheritableThreadLocal<>();

  26.     /**
  27.      * Construct a new instance.
  28.      */
  29.     public DemuxOutputStream() {
  30.         // empty
  31.     }

  32.     /**
  33.      * Binds the specified stream to the current thread.
  34.      *
  35.      * @param output
  36.      *            the stream to bind
  37.      * @return the OutputStream that was previously active
  38.      */
  39.     public OutputStream bindStream(final OutputStream output) {
  40.         final OutputStream stream = outputStreamThreadLocal.get();
  41.         outputStreamThreadLocal.set(output);
  42.         return stream;
  43.     }

  44.     /**
  45.      * Closes stream associated with current thread.
  46.      *
  47.      * @throws IOException
  48.      *             if an error occurs
  49.      */
  50.     @SuppressWarnings("resource") // we actually close the stream here
  51.     @Override
  52.     public void close() throws IOException {
  53.         IOUtils.close(outputStreamThreadLocal.get());
  54.     }

  55.     /**
  56.      * Flushes stream associated with current thread.
  57.      *
  58.      * @throws IOException
  59.      *             if an error occurs
  60.      */
  61.     @Override
  62.     public void flush() throws IOException {
  63.         @SuppressWarnings("resource")
  64.         final OutputStream output = outputStreamThreadLocal.get();
  65.         if (null != output) {
  66.             output.flush();
  67.         }
  68.     }

  69.     /**
  70.      * Writes byte to stream associated with current thread.
  71.      *
  72.      * @param ch
  73.      *            the byte to write to stream
  74.      * @throws IOException
  75.      *             if an error occurs
  76.      */
  77.     @Override
  78.     public void write(final int ch) throws IOException {
  79.         @SuppressWarnings("resource")
  80.         final OutputStream output = outputStreamThreadLocal.get();
  81.         if (null != output) {
  82.             output.write(ch);
  83.         }
  84.     }
  85. }