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

org.fusesource.jansi.io.AnsiOutputStream Maven / Gradle / Ivy

/*
 * Copyright (C) 2009-2020 the original author(s).
 *
 * Licensed 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.fusesource.jansi.io;

import java.io.FilterOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.charset.Charset;
import java.util.ArrayList;

import org.fusesource.jansi.AnsiColors;
import org.fusesource.jansi.AnsiMode;
import org.fusesource.jansi.AnsiType;

/**
 * A ANSI print stream extracts ANSI escape codes written to 
 * an output stream and calls corresponding AnsiProcessor.process* methods.
 * This particular class is not synchronized for improved performances.
 *
 * 

For more information about ANSI escape codes, see * Wikipedia article * * @author Guillaume Nodet * @since 1.0 * @see AnsiProcessor */ public class AnsiOutputStream extends FilterOutputStream { public static final byte[] RESET_CODE = "\033[0m".getBytes(); public interface IoRunnable { void run() throws IOException; } public interface WidthSupplier { int getTerminalWidth(); } public static class ZeroWidthSupplier implements WidthSupplier { @Override public int getTerminalWidth() { return 0; } } private static final int LOOKING_FOR_FIRST_ESC_CHAR = 0; private static final int LOOKING_FOR_SECOND_ESC_CHAR = 1; private static final int LOOKING_FOR_NEXT_ARG = 2; private static final int LOOKING_FOR_STR_ARG_END = 3; private static final int LOOKING_FOR_INT_ARG_END = 4; private static final int LOOKING_FOR_OSC_COMMAND = 5; private static final int LOOKING_FOR_OSC_COMMAND_END = 6; private static final int LOOKING_FOR_OSC_PARAM = 7; private static final int LOOKING_FOR_ST = 8; private static final int LOOKING_FOR_CHARSET = 9; private static final int FIRST_ESC_CHAR = 27; private static final int SECOND_ESC_CHAR = '['; private static final int SECOND_OSC_CHAR = ']'; private static final int BEL = 7; private static final int SECOND_ST_CHAR = '\\'; private static final int SECOND_CHARSET0_CHAR = '('; private static final int SECOND_CHARSET1_CHAR = ')'; private AnsiProcessor ap; private final static int MAX_ESCAPE_SEQUENCE_LENGTH = 100; private final byte[] buffer = new byte[MAX_ESCAPE_SEQUENCE_LENGTH]; private int pos = 0; private int startOfValue; private final ArrayList options = new ArrayList(); private int state = LOOKING_FOR_FIRST_ESC_CHAR; private final Charset cs; private final WidthSupplier width; private final AnsiProcessor processor; private final AnsiType type; private final AnsiColors colors; private final IoRunnable installer; private final IoRunnable uninstaller; private AnsiMode mode; private boolean resetAtUninstall; public AnsiOutputStream(OutputStream os, WidthSupplier width, AnsiMode mode, AnsiProcessor processor, AnsiType type, AnsiColors colors, Charset cs, IoRunnable installer, IoRunnable uninstaller, boolean resetAtUninstall) { super(os); this.width = width; this.processor = processor; this.type = type; this.colors = colors; this.installer = installer; this.uninstaller = uninstaller; this.resetAtUninstall = resetAtUninstall; this.cs = cs; setMode(mode); } public int getTerminalWidth() { return width.getTerminalWidth(); } public AnsiType getType() { return type; } public AnsiColors getColors() { return colors; } public AnsiMode getMode() { return mode; } public void setMode(AnsiMode mode) { ap = mode == AnsiMode.Strip ? new AnsiProcessor(out) : mode == AnsiMode.Force || processor == null ? new ColorsAnsiProcessor(out, colors) : processor; this.mode = mode; } public boolean isResetAtUninstall() { return resetAtUninstall; } public void setResetAtUninstall(boolean resetAtUninstall) { this.resetAtUninstall = resetAtUninstall; } /** * {@inheritDoc} */ @Override public void write(int data) throws IOException { switch (state) { case LOOKING_FOR_FIRST_ESC_CHAR: if (data == FIRST_ESC_CHAR) { buffer[pos++] = (byte) data; state = LOOKING_FOR_SECOND_ESC_CHAR; } else { out.write(data); } break; case LOOKING_FOR_SECOND_ESC_CHAR: buffer[pos++] = (byte) data; if (data == SECOND_ESC_CHAR) { state = LOOKING_FOR_NEXT_ARG; } else if (data == SECOND_OSC_CHAR) { state = LOOKING_FOR_OSC_COMMAND; } else if (data == SECOND_CHARSET0_CHAR) { options.add(0); state = LOOKING_FOR_CHARSET; } else if (data == SECOND_CHARSET1_CHAR) { options.add(1); state = LOOKING_FOR_CHARSET; } else { reset(false); } break; case LOOKING_FOR_NEXT_ARG: buffer[pos++] = (byte) data; if ('"' == data) { startOfValue = pos - 1; state = LOOKING_FOR_STR_ARG_END; } else if ('0' <= data && data <= '9') { startOfValue = pos - 1; state = LOOKING_FOR_INT_ARG_END; } else if (';' == data) { options.add(null); } else if ('?' == data) { options.add('?'); } else if ('=' == data) { options.add('='); } else { processEscapeCommand(data); } break; default: break; case LOOKING_FOR_INT_ARG_END: buffer[pos++] = (byte) data; if (!('0' <= data && data <= '9')) { String strValue = new String(buffer, startOfValue, (pos - 1) - startOfValue); Integer value = Integer.valueOf(strValue); options.add(value); if (data == ';') { state = LOOKING_FOR_NEXT_ARG; } else { processEscapeCommand(data); } } break; case LOOKING_FOR_STR_ARG_END: buffer[pos++] = (byte) data; if ('"' != data) { String value = new String(buffer, startOfValue, (pos - 1) - startOfValue, cs); options.add(value); if (data == ';') { state = LOOKING_FOR_NEXT_ARG; } else { processEscapeCommand(data); } } break; case LOOKING_FOR_OSC_COMMAND: buffer[pos++] = (byte) data; if ('0' <= data && data <= '9') { startOfValue = pos - 1; state = LOOKING_FOR_OSC_COMMAND_END; } else { reset(false); } break; case LOOKING_FOR_OSC_COMMAND_END: buffer[pos++] = (byte) data; if (';' == data) { String strValue = new String(buffer, startOfValue, (pos - 1) - startOfValue); Integer value = Integer.valueOf(strValue); options.add(value); startOfValue = pos; state = LOOKING_FOR_OSC_PARAM; } else if ('0' <= data && data <= '9') { // already pushed digit to buffer, just keep looking } else { // oops, did not expect this reset(false); } break; case LOOKING_FOR_OSC_PARAM: buffer[pos++] = (byte) data; if (BEL == data) { String value = new String(buffer, startOfValue, (pos - 1) - startOfValue, cs); options.add(value); processOperatingSystemCommand(); } else if (FIRST_ESC_CHAR == data) { state = LOOKING_FOR_ST; } else { // just keep looking while adding text } break; case LOOKING_FOR_ST: buffer[pos++] = (byte) data; if (SECOND_ST_CHAR == data) { String value = new String(buffer, startOfValue, (pos - 2) - startOfValue, cs); options.add(value); processOperatingSystemCommand(); } else { state = LOOKING_FOR_OSC_PARAM; } break; case LOOKING_FOR_CHARSET: options.add((char) data); processCharsetSelect(); break; } // Is it just too long? if (pos >= buffer.length) { reset(false); } } private void processCharsetSelect() throws IOException { try { reset(ap != null && ap.processCharsetSelect(options)); } catch (RuntimeException e) { reset(true); throw e; } } private void processOperatingSystemCommand() throws IOException { try { reset(ap != null && ap.processOperatingSystemCommand(options)); } catch (RuntimeException e) { reset(true); throw e; } } private void processEscapeCommand(int data) throws IOException { try { reset(ap != null && ap.processEscapeCommand(options, data)); } catch (RuntimeException e) { reset(true); throw e; } } /** * Resets all state to continue with regular parsing * @param skipBuffer if current buffer should be skipped or written to out * @throws IOException */ private void reset(boolean skipBuffer) throws IOException { if (!skipBuffer) { out.write(buffer, 0, pos); } pos = 0; startOfValue = 0; options.clear(); state = LOOKING_FOR_FIRST_ESC_CHAR; } public void install() throws IOException { if (installer != null) { installer.run(); } } public void uninstall() throws IOException { if (resetAtUninstall && type != AnsiType.Redirected && type != AnsiType.Unsupported) { setMode(AnsiMode.Default); write(RESET_CODE); flush(); } if (uninstaller != null) { uninstaller.run(); } } @Override public void close() throws IOException { uninstall(); super.close(); } }