All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.brooklyn.util.stream.LoggingOutputStream Maven / Gradle / Ivy

Go to download

Utility classes and methods developed for Brooklyn but not dependendent on Brooklyn or much else

There is a newer version: 1.1.0
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 */
package org.apache.brooklyn.util.stream;

import java.io.FilterOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.atomic.AtomicBoolean;

import org.apache.http.util.ByteArrayBuffer;
import org.slf4j.Logger;

/**
 * Wraps another output stream, intercepting the writes to log it.
 * 
 * This is not thread safe. It assumes that calls like write, flush and close 
 * will be done by the same thread.
 */
public class LoggingOutputStream extends FilterOutputStream {

    private static final OutputStream NOOP_OUTPUT_STREAM = new FilterOutputStream(null) {
        @Override public void write(int b) throws IOException {
        }
        @Override public void flush() throws IOException {
        }
        @Override public void close() throws IOException {
        }        
    };
    
    public static Builder builder() {
        return new Builder();
    }
    
    public static class Builder {
        OutputStream out;
        Logger log;
        String logPrefix;
        
        public Builder outputStream(OutputStream val) {
            this.out = val;
            return this;
        }
        public Builder logger(Logger val) {
            this.log = val;
            return this;
        }
        public Builder logPrefix(String val) {
            this.logPrefix = val;
            return this;
        }
        public LoggingOutputStream build() {
            return new LoggingOutputStream(this);
        }
    }
    
    protected final Logger log;
    protected final String logPrefix;
    private final AtomicBoolean running = new AtomicBoolean(true);
    
    // Uses byte array, rather than StringBuilder, to handle Unicode chars longer than one byte
    private ByteArrayBuffer lineSoFar = new ByteArrayBuffer(16);

    private LoggingOutputStream(Builder builder) {
        super(builder.out != null ? builder.out : NOOP_OUTPUT_STREAM);
        log = builder.log;
        logPrefix = (builder.logPrefix != null) ? builder.logPrefix : "";
      }

    @Override
    public void write(int b) throws IOException {
        if (running.get()) onChar(b);
        out.write(b);
    }

    @Override
    public void flush() throws IOException {
        try {
            if (lineSoFar.length() > 0) {
                onLine(lineSoFar.buffer(), lineSoFar.length());
                clearLineSoFar();
            }
        } finally {
            super.flush();
        }
    }
    
    // Overriding close() because FilterOutputStream's close() method pre-JDK8 has bad behavior:
    // it silently ignores any exception thrown by flush(). Instead, just close the delegate stream.
    // It should flush itself if necessary.
    @Override
    public void close() throws IOException {
        try {
            onLine(lineSoFar.buffer(), lineSoFar.length());
            clearLineSoFar();
        } finally {
            out.close();
            running.set(false);
        }
    }
    
    public void onChar(int c) {
        if (c=='\n' || c=='\r') {
            if (lineSoFar.length() > 0) {
                //suppress blank lines, so that we can treat either newline char as a line separator
                //(eg to show curl updates frequently)
                onLine(lineSoFar.buffer(), lineSoFar.length());
                clearLineSoFar();
            }
            
        } else {
            lineSoFar.append(c);
        }
    }
    
    private void clearLineSoFar() {
        lineSoFar.setLength(0);
        
        // Avoid keeping hold of a lot of memory for too long a time:
        // if we'd constructed a huge buffer, then get rid of it.
        if (lineSoFar.capacity() > 1024) {
            lineSoFar = new ByteArrayBuffer(16);
        }
    }

    public void onLine(byte[] line, int length) {
        onLine(new String(line, 0, length, StandardCharsets.UTF_8));
    }
    
    public void onLine(String line) {
        //right trim, in case there is \r or other funnies
        while (line.length()>0 && Character.isWhitespace(line.charAt(line.length()-1)))
            line = line.substring(0, line.length()-1);
        //left trim, in case there is \r or other funnies
        while (line.length()>0 && (line.charAt(0)=='\n' || line.charAt(0)=='\r'))
            line = line.substring(1);
        if (!line.isEmpty()) {
            if (log!=null && log.isDebugEnabled()) log.debug(logPrefix+line);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy