/* * Copyright (C) 2017 The Android Open Source Project * * 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 com.android.apksig.util; import static org.junit.Assert.assertEquals; import static org.junit.Assert.fail; import java.io.Closeable; import java.io.IOException; import java.nio.ByteBuffer; import java.nio.charset.StandardCharsets; import org.junit.Test; /** * Base class for testing implementations of {@link DataSink}. This class tests the contract of * {@code DataSink}. * *

To subclass, provide an implementation of {@link #createDataSink()} which returns the * implementation of {@code DataSink} you want to test. */ public abstract class DataSinkTestBase { /** * Returns a new {@link DataSink}. */ protected abstract CloseableWithDataSink createDataSink() throws IOException; /** * Returns the contents of the data sink. */ protected abstract ByteBuffer getContents(T dataSink) throws IOException; @Test public void testConsumeFromArray() throws Exception { try (CloseableWithDataSink c = createDataSink()) { T sink = c.getDataSink(); byte[] input = "abcdefg".getBytes(StandardCharsets.UTF_8); sink.consume(input, 2, 3); // "cde" sink.consume(input, 0, 1); // "a" assertContentsEquals("cdea", sink); // Zero-length chunks sink.consume(input, 0, 0); sink.consume(input, 1, 0); sink.consume(input, input.length - 2, 0); sink.consume(input, input.length - 1, 0); sink.consume(input, input.length, 0); // Invalid chunks assertConsumeArrayThrowsIOOB(sink, input, -1, 0); assertConsumeArrayThrowsIOOB(sink, input, -1, 3); assertConsumeArrayThrowsIOOB(sink, input, 0, input.length + 1); assertConsumeArrayThrowsIOOB(sink, input, input.length - 2, 4); assertConsumeArrayThrowsIOOB(sink, input, input.length + 1, 0); assertConsumeArrayThrowsIOOB(sink, input, input.length + 1, 1); assertContentsEquals("cdea", sink); } } @Test public void testConsumeFromByteBuffer() throws Exception { try (CloseableWithDataSink c = createDataSink()) { T sink = c.getDataSink(); ByteBuffer input = ByteBuffer.wrap("abcdefg".getBytes(StandardCharsets.UTF_8)); input.position(2); input.limit(5); sink.consume(input); // "cde" assertEquals(5, input.position()); assertEquals(5, input.limit()); input.position(0); input.limit(1); sink.consume(input); // "a" assertContentsEquals("cdea", sink); // Empty input sink.consume(input); assertContentsEquals("cdea", sink); // ByteBuffer which isn't backed by a byte[] input = ByteBuffer.allocateDirect(2); input.put((byte) 'X'); input.put((byte) 'Z'); input.flip(); sink.consume(input); assertContentsEquals("cdeaXZ", sink); assertEquals(2, input.position()); assertEquals(2, input.limit()); // Empty input sink.consume(input); assertContentsEquals("cdeaXZ", sink); } } /** * Returns the contents of the provided buffer as a string. The buffer's position and limit * remain unchanged. */ private static String toString(ByteBuffer buf) { return DataSourceTestBase.toString(buf); } private void assertContentsEquals(String expectedContents, T sink) throws IOException { ByteBuffer actual = getContents(sink); assertEquals(expectedContents, toString(actual)); } private static void assertConsumeArrayThrowsIOOB( DataSink sink, byte[] arr, int offset, int length) throws IOException { try { sink.consume(arr, offset, length); fail(); } catch (IndexOutOfBoundsException expected) {} } public static class CloseableWithDataSink implements Closeable { private final T mDataSink; private final Closeable mCloseable; private CloseableWithDataSink(T dataSink, Closeable closeable) { mDataSink = dataSink; mCloseable = closeable; } public static CloseableWithDataSink of(T dataSink) { return new CloseableWithDataSink<>(dataSink, null); } public static CloseableWithDataSink of( T dataSink, Closeable closeable) { return new CloseableWithDataSink<>(dataSink, closeable); } public T getDataSink() { return mDataSink; } public Closeable getCloseable() { return mCloseable; } @Override public void close() throws IOException { if (mCloseable != null) { mCloseable.close(); } } } }