001/**
002 * Logback: the reliable, generic, fast and flexible logging framework.
003 * Copyright (C) 1999-2015, QOS.ch. All rights reserved.
004 *
005 * This program and the accompanying materials are dual-licensed under
006 * either the terms of the Eclipse Public License v1.0 as published by
007 * the Eclipse Foundation
008 *
009 *   or (per the licensee's choosing)
010 *
011 * under the terms of the GNU Lesser General Public License version 2.1
012 * as published by the Free Software Foundation.
013 */
014package ch.qos.logback.access.common.servlet;
015
016import java.io.ByteArrayOutputStream;
017import java.io.IOException;
018
019import jakarta.servlet.ServletOutputStream;
020import jakarta.servlet.ServletResponse;
021import jakarta.servlet.WriteListener;
022
023public class TeeServletOutputStream extends ServletOutputStream {
024
025    final ServletOutputStream underlyingStream;
026    final ByteArrayOutputStream baosCopy;
027
028    TeeServletOutputStream(ServletResponse httpServletResponse) throws IOException {
029        // System.out.println("TeeServletOutputStream.constructor() called");
030        this.underlyingStream = httpServletResponse.getOutputStream();
031        baosCopy = new ByteArrayOutputStream();
032    }
033
034    byte[] getOutputStreamAsByteArray() {
035        return baosCopy.toByteArray();
036    }
037
038    @Override
039    public void write(int val) throws IOException {
040        if (underlyingStream != null) {
041            underlyingStream.write(val);
042            baosCopy.write(val);
043        }
044    }
045
046    @Override
047    public void write(byte[] byteArray) throws IOException {
048        if (underlyingStream == null) {
049            return;
050        }
051        // System.out.println("WRITE TeeServletOutputStream.write(byte[]) called");
052        write(byteArray, 0, byteArray.length);
053    }
054
055    @Override
056    public void write(byte byteArray[], int offset, int length) throws IOException {
057        if (underlyingStream == null) {
058            return;
059        }
060        // System.out.println("WRITE TeeServletOutputStream.write(byte[], int, int)
061        // called");
062        // System.out.println(new String(byteArray, offset, length));
063        underlyingStream.write(byteArray, offset, length);
064        baosCopy.write(byteArray, offset, length);
065    }
066
067    @Override
068    public void close() throws IOException {
069        // System.out.println("CLOSE TeeServletOutputStream.close() called");
070
071        // If the servlet accessing the stream is using a writer instead of
072        // an OutputStream, it will probably call os.close() before calling
073        // writer.close. Thus, the underlying output stream will be called
074        // before the data sent to the writer could be flushed.
075    }
076
077    @Override
078    public void flush() throws IOException {
079        if (underlyingStream == null) {
080            return;
081        }
082        // System.out.println("FLUSH TeeServletOutputStream.flush() called");
083        underlyingStream.flush();
084        baosCopy.flush();
085    }
086
087    @Override
088    public boolean isReady() {
089        throw new RuntimeException("Not yet implemented");
090    }
091
092    @Override
093    public void setWriteListener(WriteListener listener) {
094        throw new RuntimeException("Not yet implemented");
095    }
096}