001/*
002 * Copyright (C) 2022 - 2025, the original author or authors.
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 *    http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 * See the License for the specific language governing permissions and
014 * limitations under the License.
015 */
016package io.github.ascopes.jct.diagnostics;
017
018import static java.util.Objects.requireNonNull;
019
020import java.io.IOException;
021import java.io.OutputStream;
022import java.io.OutputStreamWriter;
023import java.io.Writer;
024import java.nio.charset.Charset;
025import java.util.concurrent.locks.Lock;
026import java.util.concurrent.locks.ReentrantLock;
027
028/**
029 * A writer that wraps an output stream and also writes any content to an in-memory buffer.
030 *
031 * <p>This is thread-safe.
032 *
033 * @author Ashley Scopes
034 * @since 0.0.1
035 */
036public final class TeeWriter extends Writer {
037
038  private final Lock lock;
039
040  private volatile boolean closed;
041
042  private final Writer writer;
043
044  // We use a StringBuilder and manually synchronise it rather than
045  // a string buffer, as we want to manually synchronise the builder
046  // and the delegated output writer at the same time.
047  private final StringBuilder builder;
048
049  /**
050   * Initialise the writer.
051   *
052   * @param writer the underlying writer to "tee" to.
053   */
054  public TeeWriter(Writer writer) {
055    lock = new ReentrantLock();
056    closed = false;
057
058    this.writer = requireNonNull(writer, "writer");
059    builder = new StringBuilder();
060    builder.ensureCapacity(64);
061  }
062
063  @Override
064  public void close() throws IOException {
065    // release to set and acquire to check ensures in-order operations to prevent
066    // a very minute chance of a race condition.
067    lock.lock();
068    try {
069      if (!closed) {
070        closed = true;
071        builder.trimToSize();
072        writer.flush();
073        writer.close();
074      }
075    } finally {
076      lock.unlock();
077    }
078  }
079
080  @Override
081  public void flush() throws IOException {
082    lock.lock();
083    try {
084      ensureOpen();
085      writer.flush();
086    } finally {
087      lock.unlock();
088    }
089  }
090
091  /**
092   * Get the content of the internal buffer.
093   *
094   * @return the content.
095   * @since 0.2.1
096   */
097  public String getContent() {
098    lock.lock();
099    try {
100      return builder.toString();
101    } finally {
102      lock.unlock();
103    }
104  }
105
106  /**
107   * Get the content of the internal buffer.
108   *
109   * @return the content.
110   * @since 0.2.1
111   */
112  @Override
113  public String toString() {
114    return getContent();
115  }
116
117  @Override
118  public void write(char[] buffer, int offset, int length) throws IOException {
119    lock.lock();
120    try {
121      ensureOpen();
122
123      writer.write(buffer, offset, length);
124      // Only append to the buffer once we know that the writing
125      // operation has completed.
126      builder.append(buffer, offset, length);
127    } finally {
128      lock.unlock();
129    }
130  }
131
132  private void ensureOpen() throws IOException {
133    if (closed) {
134      throw new IOException("TeeWriter is closed");
135    }
136  }
137
138  /**
139   * Create a tee writer for the given output stream.
140   *
141   * <p>Remember you may need to manually flush the tee writer for all contents to be committed to
142   * the output stream.
143   *
144   * @param outputStream the output stream.
145   * @param charset      the charset.
146   * @return the Tee Writer.
147   * @since 0.2.1
148   */
149  public static TeeWriter wrapOutputStream(OutputStream outputStream, Charset charset) {
150    requireNonNull(outputStream, "outputStream");
151    requireNonNull(charset, "charset");
152    var writer = new OutputStreamWriter(outputStream, charset);
153    return new TeeWriter(writer);
154  }
155}