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

org.conscrypt.testing.TestServer Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 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 org.conscrypt.testing;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import javax.net.ssl.SSLServerSocket;
import javax.net.ssl.SSLSocket;

/**
 * A simple socket-based test server.
 */
public final class TestServer {
    /**
     * A processor for receipt of a single message.
     */
    public interface MessageProcessor { void processMessage(byte[] message, int numBytes, OutputStream os); }

    /**
     * A {@link MessageProcessor} that simply echos back the received message to the client.
     */
    public static final class EchoProcessor implements MessageProcessor {
        @Override
        public void processMessage(byte[] message, int numBytes, OutputStream os) {
            try {
                os.write(message, 0, numBytes);
                os.flush();
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
    }

    private final SSLServerSocket serverSocket;
    private final int messageSize;
    private final byte[] buffer;
    private SSLSocket socket;
    private ExecutorService executor;
    private InputStream inputStream;
    private OutputStream outputStream;
    private volatile boolean stopping;
    private volatile MessageProcessor messageProcessor = new EchoProcessor();

    public TestServer(SSLServerSocket serverSocket, int messageSize) {
        this.serverSocket = serverSocket;
        this.messageSize = messageSize;
        buffer = new byte[messageSize];
    }

    public void setMessageProcessor(MessageProcessor messageProcessor) {
        this.messageProcessor = messageProcessor;
    }

    public Future start() {
        executor = Executors.newSingleThreadExecutor();
        return executor.submit(new AcceptTask());
    }

    public void stop() {
        try {
            stopping = true;

            if (socket != null) {
                socket.close();
                socket = null;
            }
            serverSocket.close();

            if (executor != null) {
                executor.shutdown();
                executor.awaitTermination(5, TimeUnit.SECONDS);
                executor = null;
            }
        } catch (IOException | InterruptedException e) {
            throw new RuntimeException(e);
        }
    }

    public int port() {
        return serverSocket.getLocalPort();
    }

    private final class AcceptTask implements Runnable {
        @Override
        public void run() {
            try {
                if (stopping) {
                    return;
                }
                socket = (SSLSocket) serverSocket.accept();
                inputStream = socket.getInputStream();
                outputStream = socket.getOutputStream();

                if (stopping) {
                    return;
                }
                executor.execute(new ProcessTask());
            } catch (IOException e) {
                throw new RuntimeException(e);
            }
        }
    }

    private final class ProcessTask implements Runnable {
        @Override
        public void run() {
            try {
                Thread thread = Thread.currentThread();
                while (!stopping && !thread.isInterrupted()) {
                    int bytesRead = readMessage();
                    if (!stopping && !thread.isInterrupted()) {
                        messageProcessor.processMessage(buffer, bytesRead, outputStream);
                    }
                }
            } catch (Throwable e) {
                throw new RuntimeException(e);
            }
        }

        private int readMessage() throws IOException {
            int totalBytesRead = 0;
            while (totalBytesRead < messageSize) {
                int remaining = messageSize - totalBytesRead;
                int bytesRead = inputStream.read(buffer, totalBytesRead, remaining);
                if (bytesRead == -1) {
                    break;
                }
                totalBytesRead += bytesRead;
            }
            return totalBytesRead;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy