com.hypirion.io.PipeTest.java Source code

Java tutorial

Introduction

Here is the source code for com.hypirion.io.PipeTest.java

Source

/*
 * Copyright (c) 2013 Jean Niklas L'orange. All rights reserved.
 *
 * The use and distribution terms for this software are covered by the
 * Eclipse Public License 1.0 (http://opensource.org/licenses/eclipse-1.0.php)
 * which can be found in the file LICENSE at the root of this distribution.
 *
 * By using this software in any fashion, you are agreeing to be bound by
 * the terms of this license.
 *
 * You must not remove this notice, or any other, from this software.
 */

package com.hypirion.io;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.Reader;
import java.io.Writer;
import java.io.StringWriter;
import java.io.StringReader;

import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.RandomStringUtils;

import com.hypirion.io.Pipe;

import org.junit.Test;
import static org.junit.Assert.*;

public class PipeTest {

    /**
     * Tests that an InputStream with random ascii characters will be completely
     * piped through the pipe.
     */
    @Test(timeout = 1000)
    public void testBasicStreamCapabilities() throws Exception {
        String input = RandomStringUtils.random(3708);
        InputStream in = IOUtils.toInputStream(input, "UTF-8");
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        Pipe p = new Pipe(in, out);
        p.start();
        p.join();
        in.close();
        String output = out.toString("UTF-8");
        out.close();
        assertEquals(input, output);
    }

    /**
     * Test that multiple InputStreams with random ascii characters will be
     * completely piped through the pipe, and not close the OutputStream.
     */
    @Test(timeout = 1000)
    public void testStreamConcatenation() throws Exception {
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        String input = "";
        for (int i = 0; i < 10; i++) {
            String inString = RandomStringUtils.random(3708);
            input += inString;
            InputStream in = IOUtils.toInputStream(inString, "UTF-8");
            Pipe p = new Pipe(in, out);
            p.start();
            p.join();
            in.close();
        }
        String output = out.toString("UTF-8");
        out.close();
        assertEquals(input, output);
    }

    /**
     * Test that multiple pipes running concurrently won't leave any chars
     * behind.
     */
    @Test(timeout = 1000)
    public void testConcurrentStreamPiping() throws Exception {
        final int charCount = 30;
        final char[] vals = "123456789".toCharArray();
        final int n = vals.length;
        InputStream[] istreams = new InputStream[n];

        // Generate n istreams with charCount equal elements in them.
        for (int i = 0; i < n; i++) {
            String s = "";
            for (int j = 0; j < charCount; j++) {
                s += vals[i];
            }
            InputStream is = IOUtils.toInputStream(s, "UTF-8");
            istreams[i] = new SlowInputStream(is);
        }
        ByteArrayOutputStream wrt = new ByteArrayOutputStream();
        Pipe[] pipes = new Pipe[n];
        for (int i = 0; i < n; i++) {
            pipes[i] = new Pipe(istreams[i], wrt);
        }
        for (int i = 0; i < n; i++) {
            pipes[i].start();
        }
        for (int i = 0; i < n; i++) {
            pipes[i].join();
            istreams[i].close();
        }

        // Count up elements and ensure that we've got the correct amount of
        // characters of each type.
        String out = wrt.toString("UTF-8");
        char[] output = out.toCharArray();
        wrt.close();
        for (char v : vals) {
            int sum = 0;
            for (char c : output) {
                if (c == v) {
                    sum++;
                }
            }
            assertEquals(charCount, sum);
        }
    }

    /**
     * Test that basic reader/writer capabilities work as expected.
     */
    @Test(timeout = 1000)
    public void testBasicReaderCapabilities() throws Exception {
        String input = RandomStringUtils.random(3708);
        StringReader rdr = new StringReader(input);
        StringWriter wrt = new StringWriter();
        Pipe p = new Pipe(rdr, wrt);
        p.start();
        p.join();
        String output = wrt.toString();
        rdr.close();
        wrt.close();
        assertEquals(input, output);
    }

    /**
     * Test that reading from multiple readers doesn't change or stop the
     * writer.
     */
    @Test(timeout = 1000)
    public void testReaderConcatenation() throws Exception {
        String input = "";
        StringWriter wrt = new StringWriter();
        for (int i = 0; i < 10; i++) {
            String thisInput = RandomStringUtils.random(3708);
            input += thisInput;
            StringReader rdr = new StringReader(thisInput);
            Pipe p = new Pipe(rdr, wrt);
            p.start();
            p.join();
            rdr.close();
        }
        String output = wrt.toString();
        wrt.close();
        assertEquals(input, output);
    }

    /**
     * Test that multiple pipes running concurrently won't leave any chars
     * behind.
     */
    @Test(timeout = 1000)
    public void testConcurrentReaderPiping() throws Exception {
        final int charCount = 30;
        final char[] vals = "123456789".toCharArray();
        final int n = vals.length;
        Reader[] readers = new Reader[n];
        // Generate n readers with charCount equal elements in them.
        for (int i = 0; i < n; i++) {
            String s = "";
            for (int j = 0; j < charCount; j++) {
                s += vals[i];
            }
            StringReader sr = new StringReader(s);
            readers[i] = new SlowReader(sr);
        }
        StringWriter wrt = new StringWriter();
        Pipe[] pipes = new Pipe[n];
        for (int i = 0; i < n; i++) {
            pipes[i] = new Pipe(readers[i], wrt);
        }
        for (int i = 0; i < n; i++) {
            pipes[i].start();
        }
        for (int i = 0; i < n; i++) {
            pipes[i].join();
            readers[i].close();
        }

        // Count up elements and ensure that we've got the correct amount of
        // characters of each type.
        String out = wrt.toString();
        char[] output = out.toCharArray();
        wrt.close();
        for (char v : vals) {
            int sum = 0;
            for (char c : output) {
                if (c == v) {
                    sum++;
                }
            }
            assertEquals(charCount, sum);
        }
    }

    static class SlowInputStream extends InputStream {
        final InputStream is;

        public SlowInputStream(InputStream is) {
            this.is = is;
        }

        @Override
        synchronized public void close() throws IOException {
            is.close();
        }

        @Override
        synchronized public int read(byte[] bbuf, int off, int len) throws IOException {
            try {
                Thread.yield();
            } catch (Exception e) {
            }
            return is.read(bbuf, off, 1);
        }

        @Override
        synchronized public int read() throws IOException {
            try {
                Thread.yield();
            } catch (Exception e) {
            }
            return is.read();
        }
    }

    static class SlowReader extends Reader {
        final Reader r;

        public SlowReader(Reader r) {
            this.r = r;
        }

        @Override
        synchronized public void close() throws IOException {
            r.close();
        }

        @Override
        synchronized public int read(char[] cbuf, int off, int len) throws IOException {
            try {
                Thread.yield();
            } catch (Exception e) {
            }
            return r.read(cbuf, off, 1);
        }
    }
}