/** * Copyright 2014 JogAmp Community. All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, are * permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this list of * conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, this list * of conditions and the following disclaimer in the documentation and/or other materials * provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY JogAmp Community ``AS IS'' AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JogAmp Community OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * The views and conclusions contained in the software and documentation are those of the * authors and should not be interpreted as representing official policies, either expressed * or implied, of JogAmp Community. */ package com.jogamp.common.nio; import java.io.File; import java.io.IOException; import java.io.RandomAccessFile; import java.nio.channels.FileChannel; import org.junit.Assert; import org.junit.Test; import com.jogamp.junit.util.SingletonJunitCase; import org.junit.FixMethodOrder; import org.junit.runners.MethodSorters; /** * Testing {@link MappedByteBufferInputStream} and {@link MappedByteBufferOutputStream} editing functionality. */ @FixMethodOrder(MethodSorters.NAME_ASCENDING) public class TestByteBufferOutputStream extends SingletonJunitCase { static void testImpl(final String fname, final byte[] payLoad, final long payLoadOffset, final long postPayLoadFiller, final byte[] endBytes, final int sliceShift) throws IOException { testImpl(fname, payLoad, payLoadOffset, postPayLoadFiller, endBytes, sliceShift, false); testImpl(fname, payLoad, payLoadOffset, postPayLoadFiller, endBytes, sliceShift, true); } static void testImpl(final String fname, final byte[] payLoad, final long payLoadOffset, final long postPayLoadFiller, final byte[] endBytes, final int sliceShift, final boolean synchronous) throws IOException { final File file = new File(fname); file.delete(); file.createNewFile(); file.deleteOnExit(); try { final RandomAccessFile out = new RandomAccessFile(file, "rw"); final MappedByteBufferInputStream.FileResizeOp szOp = new MappedByteBufferInputStream.FileResizeOp() { @Override public void setLength(final long newSize) throws IOException { out.setLength(newSize); } }; final MappedByteBufferInputStream mis = new MappedByteBufferInputStream(out.getChannel(), FileChannel.MapMode.READ_WRITE, MappedByteBufferInputStream.CacheMode.FLUSH_PRE_SOFT, sliceShift); final MappedByteBufferOutputStream mos = mis.getOutputStream(szOp); mos.setSynchronous(synchronous); try { // resize to payLoad start and position to it mos.setLength(payLoadOffset); Assert.assertEquals(payLoadOffset, out.length()); Assert.assertEquals(payLoadOffset, mos.length()); Assert.assertEquals(0, mos.position()); // no change mos.position(payLoadOffset); Assert.assertEquals(payLoadOffset, mos.position()); // mark, write-expand payLoad mis.mark(1); mos.write(payLoad); Assert.assertEquals(payLoadOffset+payLoad.length, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); // expand + 1 mos.setLength(payLoadOffset+payLoad.length+1); Assert.assertEquals(payLoadOffset+payLoad.length+1, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); // no change // expand up-to very end, ahead of write - position to endBytes start mos.setLength(payLoadOffset+payLoad.length+postPayLoadFiller+endBytes.length); Assert.assertEquals(payLoadOffset+payLoad.length+postPayLoadFiller+endBytes.length, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length+postPayLoadFiller+endBytes.length, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); // no change mos.skip(postPayLoadFiller); Assert.assertEquals(payLoadOffset+payLoad.length+postPayLoadFiller, mos.position()); // write endBytes (no resize) mos.write(endBytes); Assert.assertEquals(payLoadOffset+payLoad.length+postPayLoadFiller+endBytes.length, mos.position()); // Reset to payLoad, read it and verify mis.reset(); Assert.assertEquals(payLoadOffset, mos.position()); Assert.assertEquals(payLoadOffset, mis.position()); final byte[] tmp = new byte[payLoad.length]; Assert.assertEquals(payLoad.length, mis.read(tmp)); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); Assert.assertEquals(payLoadOffset+payLoad.length, mis.position()); Assert.assertArrayEquals(payLoad, tmp); // Shrink to end of payLoad, mark, read >= 0, reset .. redo Assert.assertEquals(payLoadOffset+payLoad.length+postPayLoadFiller+endBytes.length, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length+postPayLoadFiller+endBytes.length, mos.length()); mos.setLength(payLoadOffset+payLoad.length+1); Assert.assertEquals(payLoadOffset+payLoad.length+1, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); mis.mark(1); Assert.assertTrue(mis.read()>=0); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.position()); mis.reset(); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); Assert.assertTrue(mis.read()>=0); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.position()); // Shrink -1, read EOS mos.setLength(payLoadOffset+payLoad.length); Assert.assertEquals(payLoadOffset+payLoad.length, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); Assert.assertEquals(-1, mis.read()); // Expand + 1, mark, read >= 0, reset .. redo mos.setLength(payLoadOffset+payLoad.length+1); Assert.assertEquals(payLoadOffset+payLoad.length+1, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); mis.mark(1); Assert.assertTrue(mis.read()>=0); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.position()); mis.reset(); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); Assert.assertTrue(mis.read()>=0); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.position()); // Shrink -1, read EOS, write-expand, reset and verify mos.setLength(payLoadOffset+payLoad.length); Assert.assertEquals(payLoadOffset+payLoad.length, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); Assert.assertEquals(-1, mis.read()); mos.write('Z'); // expand while writing .. Assert.assertEquals(payLoadOffset+payLoad.length+1, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.position()); mis.reset(); Assert.assertEquals(payLoadOffset+payLoad.length, mos.position()); Assert.assertEquals(payLoadOffset+payLoad.length, mis.position()); Assert.assertEquals('Z', mis.read()); Assert.assertEquals(payLoadOffset+payLoad.length+1, mos.position()); Assert.assertEquals(payLoadOffset+payLoad.length+1, mis.position()); // Shrink -2, shall clear mark, test reset failure mos.setLength(payLoadOffset+payLoad.length-1); Assert.assertEquals(payLoadOffset+payLoad.length-1, out.length()); Assert.assertEquals(payLoadOffset+payLoad.length-1, mos.length()); Assert.assertEquals(payLoadOffset+payLoad.length-1, mos.position()); try { mis.reset(); Assert.assertTrue(false); // shall not reach } catch( final IOException ioe ) { Assert.assertNotNull(ioe); } mis.mark(1); // ZERO file, test reset failure, read EOS, write-expand mos.setLength(0); Assert.assertEquals(0, out.length()); Assert.assertEquals(0, mos.length()); Assert.assertEquals(0, mos.position()); try { mis.reset(); Assert.assertTrue(false); // shall not reach } catch( final IOException ioe ) { Assert.assertNotNull(ioe); } Assert.assertEquals(-1, mis.read()); mos.write('Z'); // expand while writing .. Assert.assertEquals(1, out.length()); Assert.assertEquals(1, mos.length()); Assert.assertEquals(1, mos.position()); mis.position(0); Assert.assertEquals(0, mos.position()); Assert.assertEquals(0, mis.position()); Assert.assertEquals('Z', mis.read()); } finally { mos.close(); mis.close(); out.close(); } } finally { file.delete(); } } @Test public void test00() throws IOException { final int sliceShift = 13; // 8192 bytes per slice testImpl(getSimpleTestName(".")+".bin", "123456789AB".getBytes(), 0L, 0L, "EOF".getBytes(), sliceShift); } @Test public void test01() throws IOException { final int sliceShift = 13; // 8192 bytes per slice testImpl(getSimpleTestName(".")+".bin", "123456789AB".getBytes(), 9000L, 100L, "EOF".getBytes(), sliceShift); } @Test public void test02() throws IOException { final int sliceShift = 13; // 8192 bytes per slice testImpl(getSimpleTestName(".")+".bin", "123456789AB".getBytes(), 8189L, 9001L, "EOF".getBytes(), sliceShift); } @Test public void test03() throws IOException { final int sliceShift = 13; // 8192 bytes per slice testImpl(getSimpleTestName(".")+".bin", "123456789AB".getBytes(), 58189L, 109001L, "EOF".getBytes(), sliceShift); } @Test public void test10() throws IOException { final int sliceShift = 10; // 1024 bytes per slice final byte[] payLoad = new byte[4096]; for(int i=0; i<payLoad.length; i++) { payLoad[i] = (byte)('A' + i%26); } testImpl(getSimpleTestName(".")+".bin", payLoad, 0L, 0L, "EOF".getBytes(), sliceShift); } @Test public void test11() throws IOException { final int sliceShift = 10; // 1024 bytes per slice final byte[] payLoad = new byte[4096]; for(int i=0; i<payLoad.length; i++) { payLoad[i] = (byte)('A' + i%26); } testImpl(getSimpleTestName(".")+".bin", payLoad, 1030L, 99L, "EOF".getBytes(), sliceShift); } @Test public void test12() throws IOException { final int sliceShift = 10; // 1024 bytes per slice final byte[] payLoad = new byte[4096]; for(int i=0; i<payLoad.length; i++) { payLoad[i] = (byte)('A' + i%26); } testImpl(getSimpleTestName(".")+".bin", payLoad, 1021L, 1301L, "EOF".getBytes(), sliceShift); } @Test public void test13() throws IOException { final int sliceShift = 10; // 1024 bytes per slice final byte[] payLoad = new byte[4096]; for(int i=0; i<payLoad.length; i++) { payLoad[i] = (byte)('A' + i%26); } testImpl(getSimpleTestName(".")+".bin", payLoad, 3021L, 6301L, "EOF".getBytes(), sliceShift); } static boolean manualTest = false; public static void main(final String args[]) throws IOException { for(int i=0; i<args.length; i++) { if(args[i].equals("-manual")) { manualTest = true; } } final String tstname = TestByteBufferOutputStream.class.getName(); org.junit.runner.JUnitCore.main(tstname); } }