Como determino programaticamente a disponibilidade de uma porta em uma determinada máquina usando Java?
ou seja, dado um número de porta, determine se ele já está sendo usado ou não ?.
Como determino programaticamente a disponibilidade de uma porta em uma determinada máquina usando Java?
ou seja, dado um número de porta, determine se ele já está sendo usado ou não ?.
Respostas:
Esta é a implementação proveniente do projeto Apache camel :
/**
* Checks to see if a specific port is available.
*
* @param port the port to check for availability
*/
public static boolean available(int port) {
if (port < MIN_PORT_NUMBER || port > MAX_PORT_NUMBER) {
throw new IllegalArgumentException("Invalid start port: " + port);
}
ServerSocket ss = null;
DatagramSocket ds = null;
try {
ss = new ServerSocket(port);
ss.setReuseAddress(true);
ds = new DatagramSocket(port);
ds.setReuseAddress(true);
return true;
} catch (IOException e) {
} finally {
if (ds != null) {
ds.close();
}
if (ss != null) {
try {
ss.close();
} catch (IOException e) {
/* should not be thrown */
}
}
}
return false;
}
Eles também estão verificando o DatagramSocket para verificar se a porta está disponível em UDP e TCP.
Espero que isto ajude.
ServerSocket
escuta para alguma porta, é isso que ela deve fazer e retornar a ServerSocket
. Criá-lo e, em seguida, fechá-lo e devolvê-lo não garante que um subseqüente ServerSocket
possa ser criado usando essa porta. O mesmo vale para DatagramSocket
.
No Java 7, você pode usar o try-with-resource para obter um código mais compacto:
private static boolean available(int port) {
try (Socket ignored = new Socket("localhost", port)) {
return false;
} catch (IOException ignored) {
return true;
}
}
ConnectException: 'connection refused'
, sim, ele deve retornar falso. Para tempos limite, nada que pudesse retornar seria válido, pois a resposta real não é conhecida. É por isso que essa técnica é inútil para esse fim.
Parece que a partir do Java 7, a resposta de David Santamaria não funciona mais de maneira confiável. Parece que você ainda pode usar um soquete de maneira confiável para testar a conexão.
private static boolean available(int port) {
System.out.println("--------------Testing port " + port);
Socket s = null;
try {
s = new Socket("localhost", port);
// If the code makes it this far without an exception it means
// something is using the port and has responded.
System.out.println("--------------Port " + port + " is not available");
return false;
} catch (IOException e) {
System.out.println("--------------Port " + port + " is available");
return true;
} finally {
if( s != null){
try {
s.close();
} catch (IOException e) {
throw new RuntimeException("You should handle this error." , e);
}
}
}
}
Se você não está muito preocupado com o desempenho, sempre pode tentar ouvir uma porta usando a classe ServerSocket . Se ele lança uma exceção, as probabilidades estão sendo usadas.
public static boolean isAvailable(int portNr) {
boolean portFree;
try (var ignored = new ServerSocket(portNr)) {
portFree = true;
} catch (IOException e) {
portFree = false;
}
return portFree;
}
EDIT: Se tudo o que você está tentando fazer é selecionar uma porta gratuita, new ServerSocket(0)
ela encontrará uma para você.
A solução a seguir é inspirada na implementação do Spring-core do SocketUtils (licença Apache).
Comparado a outras soluções Socket(...)
, é bastante rápido (testando 1000 portas TCP em menos de um segundo):
public static boolean isTcpPortAvailable(int port) {
try (ServerSocket serverSocket = new ServerSocket()) {
// setReuseAddress(false) is required only on OSX,
// otherwise the code will not work correctly on that platform
serverSocket.setReuseAddress(false);
serverSocket.bind(new InetSocketAddress(InetAddress.getByName("localhost"), port), 1);
return true;
} catch (Exception ex) {
return false;
}
}
As soluções baseadas em soquete try / catch podem não gerar resultados precisos (o endereço do soquete é "localhost" e, em alguns casos, a porta pode ser "ocupada" não pela interface de loopback e, pelo menos no Windows, vi esse teste falhar, ou seja o prot falsamente declarado como disponível).
Há uma biblioteca legal chamada SIGAR , o seguinte código pode conectá-lo:
Sigar sigar = new Sigar();
int flags = NetFlags.CONN_TCP | NetFlags.CONN_SERVER | NetFlags.CONN_CLIENT; NetConnection[] netConnectionList = sigar.getNetConnectionList(flags);
for (NetConnection netConnection : netConnectionList) {
if ( netConnection.getLocalPort() == port )
return false;
}
return true;
Uma limpeza da resposta apontada por David Santamaria:
/**
* Check to see if a port is available.
*
* @param port
* the port to check for availability.
*/
public static boolean isPortAvailable(int port) {
try (var ss = new ServerSocket(port); var ds = new DatagramSocket(port)) {
return true;
} catch (IOException e) {
return false;
}
}
Isso ainda está sujeita a uma condição de corrida apontado por user207421 nos comentários a resposta de David Santamaria (algo poderia pegar o porto após esse método fecha o ServerSocket
e DatagramSocket
e volta).
No meu caso, ajudou a tentar conectar-se à porta - se o serviço já estiver presente, ele responderia.
try {
log.debug("{}: Checking if port open by trying to connect as a client", portNumber);
Socket sock = new Socket("localhost", portNumber);
sock.close();
log.debug("{}: Someone responding on port - seems not open", portNumber);
return false;
} catch (Exception e) {
if (e.getMessage().contains("refused")) {
return true;
}
log.error("Troubles checking if port is open", e);
throw new RuntimeException(e);
}
No meu caso, eu tive que usar a classe DatagramSocket.
boolean isPortOccupied(int port) {
DatagramSocket sock = null;
try {
sock = new DatagramSocket(port);
sock.close();
return false;
} catch (BindException ignored) {
return true;
} catch (SocketException ex) {
System.out.println(ex);
return true;
}
}
Não se esqueça de importar primeiro
import java.net.DatagramSocket;
import java.net.BindException;
import java.net.SocketException;
Eu tentei algo assim e funcionou muito bem comigo
Socket Skt;
String host = "localhost";
int i = 8983; // port no.
try {
System.out.println("Looking for "+ i);
Skt = new Socket(host, i);
System.out.println("There is a Server on port "
+ i + " of " + host);
}
catch (UnknownHostException e) {
System.out.println("Exception occured"+ e);
}
catch (IOException e) {
System.out.println("port is not used");
}