首选 Java 方式来 ping HTTP URL 以获取可用性

2022-08-31 07:06:15

我需要一个监视器类,定期检查给定的HTTP URL是否可用。我可以使用Spring TaskExecutor抽象来处理“定期”部分,所以这不是这里的主题。问题是:在java中pingURL的首选方法是什么?

以下是我当前的代码作为起点:

try {
    final URLConnection connection = new URL(url).openConnection();
    connection.connect();
    LOG.info("Service " + url + " available, yeah!");
    available = true;
} catch (final MalformedURLException e) {
    throw new IllegalStateException("Bad URL: " + url, e);
} catch (final IOException e) {
    LOG.info("Service " + url + " unavailable, oh no!", e);
    available = false;
}
  1. 这有什么好处吗(它会做我想要的吗)?
  2. 我必须以某种方式关闭连接吗?
  3. 我想这是一个请求。有没有办法发送?GETHEAD

答案 1

这有什么好处吗(它会做我想要的吗?

你可以这样做。另一种可行的方法是使用java.net.Socket

public static boolean pingHost(String host, int port, int timeout) {
    try (Socket socket = new Socket()) {
        socket.connect(new InetSocketAddress(host, port), timeout);
        return true;
    } catch (IOException e) {
        return false; // Either timeout or unreachable or failed DNS lookup.
    }
}

还有InetAddress#isReachable()

boolean reachable = InetAddress.getByName(hostname).isReachable();

但是,这不会显式测试端口 80。由于防火墙阻止了其他端口,您可能会收到漏报。


我必须以某种方式关闭连接吗?

不,您没有明确需要。它被处理并汇集在引擎盖下。


我想这是一个GET请求。有没有办法发送HEAD?

您可以将获得的转换为 HttpURLConnection,然后使用 setRequestMethod() 来设置请求方法。但是,您需要考虑到一些较差的Web应用程序或自行开发的服务器可能会返回HEAD的HTTP 405错误(即不可用,未实现,不允许),而GET工作正常。如果您打算验证链接/资源而不是域/主机,则使用 GET 更可靠。URLConnection


在我的情况下,测试服务器的可用性是不够的,我需要测试URL(webapp可能未部署)

实际上,连接主机只会通知主机是否可用,而不会通知内容是否可用。同样好的是,Web服务器已启动没有问题,但在服务器启动期间Web应用程序无法部署。但是,这通常不会导致整个服务器出现故障。您可以通过检查 HTTP 响应代码是否为 200 来确定这一点。

HttpURLConnection connection = (HttpURLConnection) new URL(url).openConnection();
connection.setRequestMethod("HEAD");
int responseCode = connection.getResponseCode();
if (responseCode != 200) {
    // Not OK.
}

// < 100 is undetermined.
// 1nn is informal (shouldn't happen on a GET/HEAD)
// 2nn is success
// 3nn is redirect
// 4nn is client error
// 5nn is server error

有关响应状态代码的更多详细信息,请参阅 RFC 2616 第 10 节。顺便说一句,如果要确定响应数据,则不需要调用。它将隐式连接。connect()

为了将来参考,这里有一个完整的实用工具方法示例,还考虑了超时:

/**
 * Pings a HTTP URL. This effectively sends a HEAD request and returns <code>true</code> if the response code is in 
 * the 200-399 range.
 * @param url The HTTP URL to be pinged.
 * @param timeout The timeout in millis for both the connection timeout and the response read timeout. Note that
 * the total timeout is effectively two times the given timeout.
 * @return <code>true</code> if the given HTTP URL has returned response code 200-399 on a HEAD request within the
 * given timeout, otherwise <code>false</code>.
 */
public static boolean pingURL(String url, int timeout) {
    url = url.replaceFirst("^https", "http"); // Otherwise an exception may be thrown on invalid SSL certificates.

    try {
        HttpURLConnection connection = (HttpURLConnection) new URL(url).openConnection();
        connection.setConnectTimeout(timeout);
        connection.setReadTimeout(timeout);
        connection.setRequestMethod("HEAD");
        int responseCode = connection.getResponseCode();
        return (200 <= responseCode && responseCode <= 399);
    } catch (IOException exception) {
        return false;
    }
}

答案 2

不要使用 URLConnection,而是通过在 URL 对象上调用 openConnection() 来使用 HttpURLConnection

然后使用getResponseCode()将在您从连接中读取后为您提供HTTP响应。

这是代码:

    HttpURLConnection connection = null;
    try {
        URL u = new URL("http://www.google.com/");
        connection = (HttpURLConnection) u.openConnection();
        connection.setRequestMethod("HEAD");
        int code = connection.getResponseCode();
        System.out.println("" + code);
        // You can determine on HTTP return code received. 200 is success.
    } catch (MalformedURLException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    } finally {
        if (connection != null) {
            connection.disconnect();
        }
    }

还要检查类似的问题 如何检查URL是否存在或返回Java的404?

希望这有帮助。