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

org.kiwiproject.net.LocalPortChecker Maven / Gradle / Ivy

Go to download

Kiwi is a utility library. We really like Google's Guava, and also use Apache Commons. But if they don't have something we need, and we think it is useful, this is where we put it.

There is a newer version: 4.5.2
Show newest version
package org.kiwiproject.net;

import static com.google.common.base.Preconditions.checkArgument;

import lombok.extern.slf4j.Slf4j;

import java.io.IOException;
import java.net.DatagramSocket;
import java.net.ServerSocket;
import java.util.OptionalInt;
import java.util.concurrent.ThreadLocalRandom;
import java.util.stream.IntStream;

/**
 * Utility to check port availability and find open ports.
 */
@Slf4j
public class LocalPortChecker {

    public static final int MAX_PORT = 65_535;

    /**
     * Check port availability. Note specifically that zero is not a valid port for the reasons described
     * here and here.
     *
     * @param port the port to check on the local machine
     * @return {@code true} if the port is available; {@code false} otherwise
     * @throws IllegalArgumentException if port is not valid
     */
    public boolean isPortAvailable(int port) {
        checkArgument(isPortFromOneToMaxInclusive(port), "Invalid port: %s", port);

        try (var serverSocket = new ServerSocket(port); var dataSocket = new DatagramSocket(port)) {
            serverSocket.setReuseAddress(true);
            dataSocket.setReuseAddress(true);
            return true;
        } catch (IOException e) {
            LOG.trace("Error occurred checking port availability for port {}", port, e);
            return false;
        }
    }

    /**
     * Find the first open port starting at the given port inclusive, i.e., if {@code port} is 1024 then it is the first
     * port checked. If no available port is found, return an empty {@link OptionalInt}.
     *
     * @param port the port to start from
     * @return an optional containing the first open port, or an empty optional
     * @throws IllegalArgumentException if port is not between 1 and {@link #MAX_PORT}
     * @implNote Since this method looks for availability of a specific port, zero is invalid because it is a reserved
     * port used to indicate a random port and is never bound directly to.
     */
    public OptionalInt findFirstOpenPortFrom(int port) {
        checkPortOneToMaxPortInclusive(port);

        var startPort = port - 1;
        return findFirstOpenPortAbove(startPort);
    }

    /**
     * Find the first open port above the given port (i.e., if port is 1024 then the first port to be
     * checked will be 1025). If no available port is found, return an empty {@link OptionalInt}.
     *
     * @param port the port to check above
     * @return an optional containing the first open port, or an empty optional
     * @throws IllegalArgumentException if port is not between 0 and ({@link #MAX_PORT} - 1)
     */
    public OptionalInt findFirstOpenPortAbove(int port) {
        checkPortZeroInclusiveToMaxPortExclusive(port);

        return IntStream.rangeClosed(port + 1, MAX_PORT)
                .filter(this::isPortAvailable)
                .findFirst();
    }

    /**
     * Find a random open port. If no available port is found, return an empty {@link OptionalInt}.
     *
     * @return an optional containing the first random port found, or an empty optional
     */
    public OptionalInt findRandomOpenPort() {
        return findRandomOpenPortAbove(0);
    }

    /**
     * Find a random open port starting at the given port. If no available port is found, return an empty
     * {@link OptionalInt}.
     *
     * @param port the port to start from (inclusive)
     * @return an optional containing the first open port, or an empty optional
     * @throws IllegalArgumentException if port is not between 1 and {@link #MAX_PORT}
     * @implNote Since this method looks for availability of a specific port, zero is invalid because it is a reserved
     * port used to indicate a random port and is never bound directly to.
     */
    public OptionalInt findRandomOpenPortFrom(int port) {
        checkPortOneToMaxPortInclusive(port);

        var startPort = port - 1;
        return findRandomOpenPortAbove(startPort);
    }

    private static void checkPortOneToMaxPortInclusive(int port) {
        checkArgument(isPortFromOneToMaxInclusive(port), "Invalid start port: %s", port);
    }

    private static boolean isPortFromOneToMaxInclusive(int port) {
        return port > 0 && port <= MAX_PORT;
    }

    /**
     * Find a random open port starting above the given port. If no available port is found, return an empty
     * {@link OptionalInt}.
     *
     * @param port the port to start from (exclusive)
     * @return an optional containing the first open port, or an empty optional
     * @throws IllegalArgumentException if port is not between 0 and ({@link #MAX_PORT} - 1)
     * @implNote Makes up to 100 attempts to find an open port before giving up.
     */
    public OptionalInt findRandomOpenPortAbove(int port) {
        checkPortZeroInclusiveToMaxPortExclusive(port);

        var minPort = port + 1;
        var maxPortBound = MAX_PORT + 1;
        return IntStream.generate(() -> ThreadLocalRandom.current().nextInt(minPort, maxPortBound))
                .filter(this::isPortAvailable)
                .limit(100)
                .findFirst();
    }

    private static void checkPortZeroInclusiveToMaxPortExclusive(int port) {
        checkArgument(port >= 0 && port < MAX_PORT, "Invalid start port: %s", port);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy