blob: a9462b29223e3e3de311bcd6ba8a07a39cb8db04 [file] [log] [blame]
/*
* 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.maven.shared.scriptinterpreter;
import java.io.File;
import java.io.IOException;
import java.io.OutputStream;
import java.io.PrintStream;
import java.nio.file.Files;
/**
* <p>FileLogger class.</p>
*/
public class FileLogger implements ExecutionLogger, AutoCloseable {
/**
* The path to the log file.
*/
private File file;
/**
* The underlying file stream this logger writes to.
*/
private PrintStream stream;
/**
* Creates a new logger that writes to the specified file.
*
* @param outputFile The path to the output file, if null all message will be discarded.
* @throws java.io.IOException If the output file could not be created.
*/
public FileLogger(File outputFile) throws IOException {
this(outputFile, null);
}
/**
* Creates a new logger that writes to the specified file and optionally mirrors messages.
*
* @param outputFile The path to the output file, if null all message will be discarded.
* @param mirrorHandler The class which handle mirrored message, can be <code>null</code>.
* @throws java.io.IOException If the output file could not be created.
*/
public FileLogger(File outputFile, FileLoggerMirrorHandler mirrorHandler) throws IOException {
this.file = outputFile;
OutputStream outputStream;
if (outputFile != null) {
outputFile.getParentFile().mkdirs();
outputStream = Files.newOutputStream(outputFile.toPath());
} else {
outputStream = new NullOutputStream();
}
if (mirrorHandler != null) {
stream = new PrintStream(new MirrorStreamWrapper(outputStream, mirrorHandler));
} else {
stream = new PrintStream(outputStream);
}
}
/**
* Gets the path to the output file.
*
* @return The path to the output file, never <code>null</code>.
*/
public File getOutputFile() {
return file;
}
/**
* Gets the underlying stream used to write message to the log file.
*
* @return The underlying stream used to write message to the log file, never <code>null</code>.
*/
@Override
public PrintStream getPrintStream() {
return stream;
}
/**
* Writes the specified line to the log file
* and invoke {@link FileLoggerMirrorHandler#consumeOutput(String)} if is given.
*
* @param line The message to log.
*/
@Override
public void consumeLine(String line) {
stream.println(line);
stream.flush();
}
/**
* Closes the underlying file stream.
*/
public void close() {
if (stream != null) {
stream.flush();
stream.close();
stream = null;
}
}
private static class MirrorStreamWrapper extends OutputStream {
private OutputStream out;
private final FileLoggerMirrorHandler mirrorHandler;
private StringBuilder lineBuffer;
MirrorStreamWrapper(OutputStream outputStream, FileLoggerMirrorHandler mirrorHandler) {
this.out = outputStream;
this.mirrorHandler = mirrorHandler;
this.lineBuffer = new StringBuilder();
}
@Override
public void write(int b) throws IOException {
out.write(b);
lineBuffer.append((char) (b));
}
@Override
public void write(byte[] b, int off, int len) throws IOException {
out.write(b, off, len);
lineBuffer.append(new String(b, off, len));
}
@Override
public void flush() throws IOException {
out.flush();
int len = lineBuffer.length();
if (len == 0) {
// nothing to log
return;
}
// remove line end for log
while (len > 0 && (lineBuffer.charAt(len - 1) == '\n' || lineBuffer.charAt(len - 1) == '\r')) {
len--;
}
lineBuffer.setLength(len);
mirrorHandler.consumeOutput(lineBuffer.toString());
// clear buffer
lineBuffer = new StringBuilder();
}
@Override
public void close() throws IOException {
flush();
if (out != null) {
out.close();
out = null;
}
}
}
private static class NullOutputStream extends OutputStream {
@Override
public void write(int b) {
// do nothing
}
@Override
public void write(byte[] b, int off, int len) {
// do nothing
}
}
}