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

com.hotels.bdp.circustrain.common.test.junit.rules.ServerSocketRule Maven / Gradle / Ivy

There is a newer version: 16.4.1
Show newest version
/**
 * Copyright (C) 2016-2017 Expedia Inc.
 *
 * 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.hotels.bdp.circustrain.common.test.junit.rules;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.InetSocketAddress;
import java.nio.channels.AsynchronousServerSocketChannel;
import java.nio.channels.AsynchronousSocketChannel;
import java.nio.channels.Channels;
import java.nio.channels.CompletionHandler;
import java.util.concurrent.Callable;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;

import org.junit.rules.ExternalResource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.google.common.io.ByteStreams;
import com.google.common.util.concurrent.MoreExecutors;

/**
 * Emulates a server that only receives packets but doesn't emit responses to any requests.
 * 

* This class can be used to emulate a Graphite Carbon relay, for example. *

*/ public class ServerSocketRule extends ExternalResource { private static final Logger LOG = LoggerFactory.getLogger(ServerSocketRule.class); private final ExecutorService executor = MoreExecutors.sameThreadExecutor(); private final AsynchronousServerSocketChannel listener; private final InetSocketAddress address; private final ByteArrayOutputStream output = new ByteArrayOutputStream(); private final ConcurrentLinkedQueue> requests = new ConcurrentLinkedQueue<>(); public ServerSocketRule() { try { listener = AsynchronousServerSocketChannel.open().bind(new InetSocketAddress(0)); address = (InetSocketAddress) listener.getLocalAddress(); } catch (IOException e) { throw new RuntimeException(e); } } @Override protected void before() throws Throwable { listener.accept(null, new CompletionHandler() { @Override public void completed(AsynchronousSocketChannel channel, Void attachment) { listener.accept(null, this); handle(channel); } @Override public void failed(Throwable exception, Void attachment) { LOG.warn("Failed to process request", exception); } }); } private void handle(final AsynchronousSocketChannel channel) { requests.offer(executor.submit(new Callable() { @Override public Void call() throws Exception { synchronized (output) { try (InputStream input = Channels.newInputStream(channel)) { ByteStreams.copy(input, output); } catch (IOException e) { throw new RuntimeException("Error processing user request", e); } } return null; } })); } @Override protected void after() { executor.shutdown(); try { executor.awaitTermination(1L, TimeUnit.SECONDS); listener.close(); } catch (InterruptedException e) { throw new RuntimeException(e); } catch (IOException e) { throw new RuntimeException(e); } } public byte[] getOutput() { synchronized (output) { return output.toByteArray(); } } public void awaitRequests(int requestCount, long timeout, TimeUnit unit) { while (requestCount > 0) { if (requests.peek() == null) { try { Thread.sleep(unit.toMillis(timeout)); } catch (InterruptedException e) { throw new RuntimeException("Interrupted whilst waiting for requests", e); } } if (requests.peek() == null) { throw new RuntimeException("No requests have been received"); } try { requests.peek().get(timeout, unit); requests.poll(); requestCount--; } catch (Exception e) { throw new RuntimeException("Error while waiting for request completion", e); } } } public int port() { return address.getPort(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy