/* * Copyright (c) 2017, 2019, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ import java.io.ByteArrayOutputStream; import java.io.FilterInputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.Serializable; import java.net.InetAddress; import java.net.ServerSocket; import java.net.Socket; import java.net.SocketAddress; import java.net.SocketException; import java.net.SocketOption; import java.nio.channels.ServerSocketChannel; import java.nio.channels.SocketChannel; import java.rmi.server.RMIClientSocketFactory; import java.rmi.server.RMIServerSocketFactory; import java.rmi.server.RMISocketFactory; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import java.util.Objects; import java.util.Set; import org.testng.Assert; import org.testng.annotations.Test; import org.testng.annotations.DataProvider; /* * @test * @summary TestSocket Factory and tests of the basic trigger, match, and replace functions * @run testng TestSocketFactory * @bug 8186539 */ /** * A RMISocketFactory utility factory to log RMI stream contents and to * trigger, and then match and replace output stream contents to simulate failures. *

* The trigger is a sequence of bytes that must be found before looking * for the bytes to match and replace. If the trigger sequence is empty * matching is immediately enabled. While waiting for the trigger to be found * bytes written to the streams are written through to the output stream. * The when triggered and when a trigger is non-empty, matching looks for * the sequence of bytes supplied. If the sequence is empty, no matching or * replacement is performed. * While waiting for a complete match, the partial matched bytes are not * written to the output stream. When the match is incomplete, the partial * matched bytes are written to the output. When a match is complete the * full replacement byte array is written to the output. *

* The trigger, match, and replacement bytes arrays can be changed at any * time and immediately reset and restart matching. Changes are propagated * to all of the sockets created from the factories immediately. */ public class TestSocketFactory extends RMISocketFactory implements RMIClientSocketFactory, RMIServerSocketFactory, Serializable { private static final long serialVersionUID = 1L; private volatile transient byte[] triggerBytes; private volatile transient byte[] matchBytes; private volatile transient byte[] replaceBytes; private transient final List sockets = new ArrayList<>(); private transient final List serverSockets = new ArrayList<>(); static final byte[] EMPTY_BYTE_ARRAY = new byte[0]; // True to enable logging of matches and replacements. private static volatile boolean debugLogging = false; /** * Debugging output can be synchronized with logging of RMI actions. * * @param format a printf format * @param args any args */ public static void DEBUG(String format, Object... args) { if (debugLogging) { System.err.printf(format, args); } } /** * Create a socket factory that creates InputStreams * and OutputStreams that log. */ public TestSocketFactory() { this.triggerBytes = EMPTY_BYTE_ARRAY; this.matchBytes = EMPTY_BYTE_ARRAY; this.replaceBytes = EMPTY_BYTE_ARRAY; } /** * Set debug to true to generate logging output of matches and substitutions. * @param debug {@code true} to generate logging output * @return the previous value */ public static boolean setDebug(boolean debug) { boolean oldDebug = debugLogging; debugLogging = debug; return oldDebug; } /** * Set the match and replacement bytes, with an empty trigger. * The match and replacements are propagated to all existing sockets. * * @param matchBytes bytes to match * @param replaceBytes bytes to replace the matched bytes */ public void setMatchReplaceBytes(byte[] matchBytes, byte[] replaceBytes) { setMatchReplaceBytes(EMPTY_BYTE_ARRAY, matchBytes, replaceBytes); } /** * Set the trigger, match, and replacement bytes. * The trigger, match, and replacements are propagated to all existing sockets. * * @param triggerBytes array of bytes to use as a trigger, may be zero length * @param matchBytes bytes to match after the trigger has been seen * @param replaceBytes bytes to replace the matched bytes */ public synchronized void setMatchReplaceBytes(byte[] triggerBytes, byte[] matchBytes, byte[] replaceBytes) { this.triggerBytes = Objects.requireNonNull(triggerBytes, "triggerBytes"); this.matchBytes = Objects.requireNonNull(matchBytes, "matchBytes"); this.replaceBytes = Objects.requireNonNull(replaceBytes, "replaceBytes"); sockets.forEach( s -> s.setMatchReplaceBytes(triggerBytes, matchBytes, replaceBytes)); serverSockets.forEach( s -> s.setMatchReplaceBytes(triggerBytes, matchBytes, replaceBytes)); } @Override public synchronized Socket createSocket(String host, int port) throws IOException { Socket socket = RMISocketFactory.getDefaultSocketFactory() .createSocket(host, port); InterposeSocket s = new InterposeSocket(socket, triggerBytes, matchBytes, replaceBytes); sockets.add(s); return s; } /** * Return the current list of sockets. * @return Return a snapshot of the current list of sockets */ public synchronized List getSockets() { List snap = new ArrayList<>(sockets); return snap; } @Override public synchronized ServerSocket createServerSocket(int port) throws IOException { ServerSocket serverSocket = RMISocketFactory.getDefaultSocketFactory() .createServerSocket(port); InterposeServerSocket ss = new InterposeServerSocket(serverSocket, triggerBytes, matchBytes, replaceBytes); serverSockets.add(ss); return ss; } /** * Return the current list of server sockets. * @return Return a snapshot of the current list of server sockets */ public synchronized List getServerSockets() { List snap = new ArrayList<>(serverSockets); return snap; } /** * An InterposeSocket wraps a socket that produces InputStreams * and OutputStreams that log the traffic. * The OutputStreams it produces watch for a trigger and then * match an array of bytes and replace them. * Useful for injecting protocol and content errors. */ public static class InterposeSocket extends Socket { private final Socket socket; private InputStream in; private MatchReplaceOutputStream out; private volatile byte[] triggerBytes; private volatile byte[] matchBytes; private volatile byte[] replaceBytes; private final ByteArrayOutputStream inLogStream; private final ByteArrayOutputStream outLogStream; private final String name; private static volatile int num = 0; // index for created Interpose509s /** * Construct a socket that interposes on a socket to match and replace. * The trigger is empty. * @param socket the underlying socket * @param matchBytes the bytes that must match * @param replaceBytes the replacement bytes */ public InterposeSocket(Socket socket, byte[] matchBytes, byte[] replaceBytes) { this(socket, EMPTY_BYTE_ARRAY, matchBytes, replaceBytes); } /** * Construct a socket that interposes on a socket to match and replace. * @param socket the underlying socket * @param triggerBytes array of bytes to enable matching * @param matchBytes the bytes that must match * @param replaceBytes the replacement bytes */ public InterposeSocket(Socket socket, byte[] triggerBytes, byte[] matchBytes, byte[] replaceBytes) { this.socket = socket; this.triggerBytes = Objects.requireNonNull(triggerBytes, "triggerBytes"); this.matchBytes = Objects.requireNonNull(matchBytes, "matchBytes"); this.replaceBytes = Objects.requireNonNull(replaceBytes, "replaceBytes"); this.inLogStream = new ByteArrayOutputStream(); this.outLogStream = new ByteArrayOutputStream(); this.name = "IS" + ++num + "::" + Thread.currentThread().getName() + ": " + socket.getLocalPort() + " < " + socket.getPort(); } /** * Set the match and replacement bytes, with an empty trigger. * The match and replacements are propagated to all existing sockets. * * @param matchBytes bytes to match * @param replaceBytes bytes to replace the matched bytes */ public void setMatchReplaceBytes(byte[] matchBytes, byte[] replaceBytes) { this.setMatchReplaceBytes(EMPTY_BYTE_ARRAY, matchBytes, replaceBytes); } /** * Set the trigger, match, and replacement bytes. * The trigger, match, and replacements are propagated to the * MatchReplaceOutputStream, if it has been created. * * @param triggerBytes array of bytes to use as a trigger, may be zero length * @param matchBytes bytes to match after the trigger has been seen * @param replaceBytes bytes to replace the matched bytes */ public synchronized void setMatchReplaceBytes(byte[] triggerBytes, byte[] matchBytes, byte[] replaceBytes) { this.triggerBytes = triggerBytes; this.matchBytes = matchBytes; this.replaceBytes = replaceBytes; if (out != null) { out.setMatchReplaceBytes(triggerBytes, matchBytes, replaceBytes); } else { DEBUG("InterposeSocket.setMatchReplaceBytes with out == null%n"); } } @Override public void connect(SocketAddress endpoint) throws IOException { socket.connect(endpoint); } @Override public void connect(SocketAddress endpoint, int timeout) throws IOException { socket.connect(endpoint, timeout); } @Override public void bind(SocketAddress bindpoint) throws IOException { socket.bind(bindpoint); } @Override public InetAddress getInetAddress() { return socket.getInetAddress(); } @Override public InetAddress getLocalAddress() { return socket.getLocalAddress(); } @Override public int getPort() { return socket.getPort(); } @Override public int getLocalPort() { return socket.getLocalPort(); } @Override public SocketAddress getRemoteSocketAddress() { return socket.getRemoteSocketAddress(); } @Override public SocketAddress getLocalSocketAddress() { return socket.getLocalSocketAddress(); } @Override public SocketChannel getChannel() { return socket.getChannel(); } @Override public synchronized void close() throws IOException { socket.close(); } @Override public String toString() { return "InterposeSocket " + name + ": " + socket.toString(); } @Override public boolean isConnected() { return socket.isConnected(); } @Override public boolean isBound() { return socket.isBound(); } @Override public boolean isClosed() { return socket.isClosed(); } @Override public Socket setOption(SocketOption name, T value) throws IOException { return socket.setOption(name, value); } @Override public T getOption(SocketOption name) throws IOException { return socket.getOption(name); } @Override public Set> supportedOptions() { return socket.supportedOptions(); } @Override public synchronized InputStream getInputStream() throws IOException { if (in == null) { in = socket.getInputStream(); String name = Thread.currentThread().getName() + ": " + socket.getLocalPort() + " < " + socket.getPort(); in = new LoggingInputStream(in, name, inLogStream); DEBUG("Created new LoggingInputStream: %s%n", name); } return in; } @Override public synchronized OutputStream getOutputStream() throws IOException { if (out == null) { OutputStream o = socket.getOutputStream(); String name = Thread.currentThread().getName() + ": " + socket.getLocalPort() + " > " + socket.getPort(); out = new MatchReplaceOutputStream(o, name, outLogStream, triggerBytes, matchBytes, replaceBytes); DEBUG("Created new MatchReplaceOutputStream: %s%n", name); } return out; } /** * Return the bytes logged from the input stream. * @return Return the bytes logged from the input stream. */ public byte[] getInLogBytes() { return inLogStream.toByteArray(); } /** * Return the bytes logged from the output stream. * @return Return the bytes logged from the output stream. */ public byte[] getOutLogBytes() { return outLogStream.toByteArray(); } } /** * InterposeServerSocket is a ServerSocket that wraps each Socket it accepts * with an InterposeSocket so that its input and output streams can be monitored. */ public static class InterposeServerSocket extends ServerSocket { private final ServerSocket socket; private volatile byte[] triggerBytes; private volatile byte[] matchBytes; private volatile byte[] replaceBytes; private final List sockets = new ArrayList<>(); /** * Construct a server socket that interposes on a socket to match and replace. * The trigger is empty. * @param socket the underlying socket * @param matchBytes the bytes that must match * @param replaceBytes the replacement bytes */ public InterposeServerSocket(ServerSocket socket, byte[] matchBytes, byte[] replaceBytes) throws IOException { this(socket, EMPTY_BYTE_ARRAY, matchBytes, replaceBytes); } /** * Construct a server socket that interposes on a socket to match and replace. * @param socket the underlying socket * @param triggerBytes array of bytes to enable matching * @param matchBytes the bytes that must match * @param replaceBytes the replacement bytes */ public InterposeServerSocket(ServerSocket socket, byte[] triggerBytes, byte[] matchBytes, byte[] replaceBytes) throws IOException { this.socket = socket; this.triggerBytes = Objects.requireNonNull(triggerBytes, "triggerBytes"); this.matchBytes = Objects.requireNonNull(matchBytes, "matchBytes"); this.replaceBytes = Objects.requireNonNull(replaceBytes, "replaceBytes"); } /** * Set the match and replacement bytes, with an empty trigger. * The match and replacements are propagated to all existing sockets. * * @param matchBytes bytes to match * @param replaceBytes bytes to replace the matched bytes */ public void setMatchReplaceBytes(byte[] matchBytes, byte[] replaceBytes) { setMatchReplaceBytes(EMPTY_BYTE_ARRAY, matchBytes, replaceBytes); } /** * Set the trigger, match, and replacement bytes. * The trigger, match, and replacements are propagated to all existing sockets. * * @param triggerBytes array of bytes to use as a trigger, may be zero length * @param matchBytes bytes to match after the trigger has been seen * @param replaceBytes bytes to replace the matched bytes */ public synchronized void setMatchReplaceBytes(byte[] triggerBytes, byte[] matchBytes, byte[] replaceBytes) { this.triggerBytes = triggerBytes; this.matchBytes = matchBytes; this.replaceBytes = replaceBytes; sockets.forEach(s -> s.setMatchReplaceBytes(triggerBytes, matchBytes, replaceBytes)); } /** * Return a snapshot of the current list of sockets created from this server socket. * @return Return a snapshot of the current list of sockets */ public synchronized List getSockets() { List snap = new ArrayList<>(sockets); return snap; } @Override public void bind(SocketAddress endpoint) throws IOException { socket.bind(endpoint); } @Override public void bind(SocketAddress endpoint, int backlog) throws IOException { socket.bind(endpoint, backlog); } @Override public InetAddress getInetAddress() { return socket.getInetAddress(); } @Override public int getLocalPort() { return socket.getLocalPort(); } @Override public SocketAddress getLocalSocketAddress() { return socket.getLocalSocketAddress(); } @Override public Socket accept() throws IOException { Socket s = socket.accept(); synchronized(this) { InterposeSocket aSocket = new InterposeSocket(s, matchBytes, replaceBytes); sockets.add(aSocket); return aSocket; } } @Override public void close() throws IOException { socket.close(); } @Override public ServerSocketChannel getChannel() { return socket.getChannel(); } @Override public boolean isClosed() { return socket.isClosed(); } @Override public String toString() { return socket.toString(); } @Override public ServerSocket setOption(SocketOption name, T value) throws IOException { return socket.setOption(name, value); } @Override public T getOption(SocketOption name) throws IOException { return socket.getOption(name); } @Override public Set> supportedOptions() { return socket.supportedOptions(); } @Override public synchronized void setSoTimeout(int timeout) throws SocketException { socket.setSoTimeout(timeout); } @Override public synchronized int getSoTimeout() throws IOException { return socket.getSoTimeout(); } } /** * LoggingInputStream is a stream and logs all bytes read to it. * For identification it is given a name. */ public static class LoggingInputStream extends FilterInputStream { private int bytesIn = 0; private final String name; private final OutputStream log; public LoggingInputStream(InputStream in, String name, OutputStream log) { super(in); this.name = name; this.log = log; } @Override public int read() throws IOException { int b = super.read(); if (b >= 0) { log.write(b); bytesIn++; } return b; } @Override public int read(byte[] b, int off, int len) throws IOException { int bytes = super.read(b, off, len); if (bytes > 0) { log.write(b, off, bytes); bytesIn += bytes; } return bytes; } @Override public int read(byte[] b) throws IOException { return read(b, 0, b.length); } @Override public void close() throws IOException { super.close(); } @Override public String toString() { return String.format("%s: In: (%d)", name, bytesIn); } } /** * An OutputStream that looks for a trigger to enable matching and * replaces one string of bytes with another. * If any range matches, the match starts after the partial match. */ static class MatchReplaceOutputStream extends OutputStream { private final OutputStream out; private final String name; private volatile byte[] triggerBytes; private volatile byte[] matchBytes; private volatile byte[] replaceBytes; int triggerIndex; int matchIndex; private int bytesOut = 0; private final OutputStream log; MatchReplaceOutputStream(OutputStream out, String name, OutputStream log, byte[] matchBytes, byte[] replaceBytes) { this(out, name, log, EMPTY_BYTE_ARRAY, matchBytes, replaceBytes); } MatchReplaceOutputStream(OutputStream out, String name, OutputStream log, byte[] triggerBytes, byte[] matchBytes, byte[] replaceBytes) { this.out = out; this.name = name; this.triggerBytes = Objects.requireNonNull(triggerBytes, "triggerBytes"); triggerIndex = 0; this.matchBytes = Objects.requireNonNull(matchBytes, "matchBytes"); this.replaceBytes = Objects.requireNonNull(replaceBytes, "replaceBytes"); matchIndex = 0; this.log = log; } public void setMatchReplaceBytes(byte[] matchBytes, byte[] replaceBytes) { setMatchReplaceBytes(EMPTY_BYTE_ARRAY, matchBytes, replaceBytes); } public void setMatchReplaceBytes(byte[] triggerBytes, byte[] matchBytes, byte[] replaceBytes) { this.triggerBytes = Objects.requireNonNull(triggerBytes, "triggerBytes"); triggerIndex = 0; this.matchBytes = Objects.requireNonNull(matchBytes, "matchBytes"); this.replaceBytes = Objects.requireNonNull(replaceBytes, "replaceBytes"); matchIndex = 0; } public void write(int b) throws IOException { b = b & 0xff; if (matchBytes.length == 0) { // fast path, no match out.write(b); log.write(b); bytesOut++; return; } // if trigger not satisfied, keep looking if (triggerBytes.length != 0 && triggerIndex < triggerBytes.length) { out.write(b); log.write(b); bytesOut++; triggerIndex = (b == (triggerBytes[triggerIndex] & 0xff)) ? ++triggerIndex // matching advance : 0; // no match, reset } else { // trigger not used or has been satisfied if (b == (matchBytes[matchIndex] & 0xff)) { if (++matchIndex >= matchBytes.length) { matchIndex = 0; triggerIndex = 0; // match/replace ok, reset trigger DEBUG("TestSocketFactory MatchReplace %s replaced %d bytes " + "at offset: %d (x%04x)%n", name, replaceBytes.length, bytesOut, bytesOut); out.write(replaceBytes); log.write(replaceBytes); bytesOut += replaceBytes.length; } } else { if (matchIndex > 0) { // mismatch, write out any that matched already DEBUG("Partial match %s matched %d bytes at offset: %d (0x%04x), " + " expected: x%02x, actual: x%02x%n", name, matchIndex, bytesOut, bytesOut, matchBytes[matchIndex], b); out.write(matchBytes, 0, matchIndex); log.write(matchBytes, 0, matchIndex); bytesOut += matchIndex; matchIndex = 0; } if (b == (matchBytes[matchIndex] & 0xff)) { matchIndex++; } else { out.write(b); log.write(b); bytesOut++; } } } } public void flush() throws IOException { if (matchIndex > 0) { // write out any that matched already to avoid consumer hang. // Match/replace across a flush is not supported. DEBUG( "Flush partial match %s matched %d bytes at offset: %d (0x%04x)%n", name, matchIndex, bytesOut, bytesOut); out.write(matchBytes, 0, matchIndex); log.write(matchBytes, 0, matchIndex); bytesOut += matchIndex; matchIndex = 0; } } @Override public String toString() { return String.format("%s: Out: (%d)", name, bytesOut); } } private static byte[] obj1Data = new byte[] { 0x7e, 0x7e, 0x7e, (byte) 0x80, 0x05, 0x7f, 0x7f, 0x7f, 0x73, 0x72, 0x00, 0x10, // TC_OBJECT, TC_CLASSDESC, length = 16 (byte)'j', (byte)'a', (byte)'v', (byte)'a', (byte)'.', (byte)'l', (byte)'a', (byte)'n', (byte)'g', (byte)'.', (byte)'n', (byte)'u', (byte)'m', (byte)'b', (byte)'e', (byte)'r' }; private static byte[] obj1Result = new byte[] { 0x7e, 0x7e, 0x7e, (byte) 0x80, 0x05, 0x7f, 0x7f, 0x7f, 0x73, 0x72, 0x00, 0x11, // TC_OBJECT, TC_CLASSDESC, length = 17 (byte)'j', (byte)'a', (byte)'v', (byte)'a', (byte)'.', (byte)'l', (byte)'a', (byte)'n', (byte)'g', (byte)'.', (byte)'I', (byte)'n', (byte)'t', (byte)'e', (byte)'g', (byte)'e', (byte)'r' }; private static byte[] obj1Trigger = new byte[] { (byte) 0x80, 0x05 }; private static byte[] obj1Trigger2 = new byte[] { 0x7D, 0x7D, 0x7D, 0x7D, }; private static byte[] obj1Trigger3 = new byte[] { 0x7F, }; private static byte[] obj1Match = new byte[] { 0x73, 0x72, 0x00, 0x10, // TC_OBJECT, TC_CLASSDESC, length = 16 (byte)'j', (byte)'a', (byte)'v', (byte)'a', (byte)'.', (byte)'l', (byte)'a', (byte)'n', (byte)'g', (byte)'.', (byte)'n', (byte)'u', (byte)'m', (byte)'b', (byte)'e', (byte)'r' }; private static byte[] obj1Repl = new byte[] { 0x73, 0x72, 0x00, 0x11, // TC_OBJECT, TC_CLASSDESC, length = 17 (byte)'j', (byte)'a', (byte)'v', (byte)'a', (byte)'.', (byte)'l', (byte)'a', (byte)'n', (byte)'g', (byte)'.', (byte)'I', (byte)'n', (byte)'t', (byte)'e', (byte)'g', (byte)'e', (byte)'r' }; @DataProvider(name = "MatchReplaceData") static Object[][] matchReplaceData() { byte[] empty = new byte[0]; byte[] byte1 = new byte[]{1, 2, 3, 4, 5, 6}; byte[] bytes2 = new byte[]{1, 2, 4, 3, 5, 6}; byte[] bytes3 = new byte[]{6, 5, 4, 3, 2, 1}; byte[] bytes4 = new byte[]{1, 2, 0x10, 0x20, 0x30, 0x40, 5, 6}; byte[] bytes4a = new byte[]{1, 2, 0x10, 0x20, 0x30, 0x40, 5, 7}; // mostly matches bytes4 byte[] bytes5 = new byte[]{0x30, 0x40, 5, 6}; byte[] bytes6 = new byte[]{1, 2, 0x10, 0x20, 0x30}; return new Object[][]{ {EMPTY_BYTE_ARRAY, new byte[]{}, new byte[]{}, empty, empty}, {EMPTY_BYTE_ARRAY, new byte[]{}, new byte[]{}, byte1, byte1}, {EMPTY_BYTE_ARRAY, new byte[]{3, 4}, new byte[]{4, 3}, byte1, bytes2}, //swap bytes {EMPTY_BYTE_ARRAY, new byte[]{3, 4}, new byte[]{0x10, 0x20, 0x30, 0x40}, byte1, bytes4}, // insert {EMPTY_BYTE_ARRAY, new byte[]{1, 2, 0x10, 0x20}, new byte[]{}, bytes4, bytes5}, // delete head {EMPTY_BYTE_ARRAY, new byte[]{0x40, 5, 6}, new byte[]{}, bytes4, bytes6}, // delete tail {EMPTY_BYTE_ARRAY, new byte[]{0x40, 0x50}, new byte[]{0x60, 0x50}, bytes4, bytes4}, // partial match, replace nothing {EMPTY_BYTE_ARRAY, bytes4a, bytes3, bytes4, bytes4}, // long partial match, not replaced {EMPTY_BYTE_ARRAY, obj1Match, obj1Repl, obj1Match, obj1Repl}, {obj1Trigger, obj1Match, obj1Repl, obj1Data, obj1Result}, {obj1Trigger3, obj1Match, obj1Repl, obj1Data, obj1Result}, // different trigger, replace {obj1Trigger2, obj1Match, obj1Repl, obj1Data, obj1Data}, // no trigger, no replace }; } @Test(dataProvider = "MatchReplaceData") public static void test1(byte[] trigger, byte[] match, byte[] replace, byte[] input, byte[] expected) { System.out.printf("trigger: %s, match: %s, replace: %s%n", Arrays.toString(trigger), Arrays.toString(match), Arrays.toString(replace)); try (ByteArrayOutputStream output = new ByteArrayOutputStream(); ByteArrayOutputStream log = new ByteArrayOutputStream(); OutputStream out = new MatchReplaceOutputStream(output, "test3", log, trigger, match, replace)) { out.write(input); byte[] actual = output.toByteArray(); long index = Arrays.mismatch(actual, expected); if (index >= 0) { System.out.printf("array mismatch, offset: %d%n", index); System.out.printf("actual: %s%n", Arrays.toString(actual)); System.out.printf("expected: %s%n", Arrays.toString(expected)); } Assert.assertEquals(actual, expected, "match/replace fail"); } catch (IOException ioe) { Assert.fail("unexpected exception", ioe); } } }