/*******************************************************************************
* This file is part of OpenNMS(R).
*
* Copyright (C) 2008-2011 The OpenNMS Group, Inc.
* OpenNMS(R) is Copyright (C) 1999-2011 The OpenNMS Group, Inc.
*
* OpenNMS(R) is a registered trademark of The OpenNMS Group, Inc.
*
* OpenNMS(R) is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published
* by the Free Software Foundation, either version 3 of the License,
* or (at your option) any later version.
*
* OpenNMS(R) is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with OpenNMS(R). If not, see:
* http://www.gnu.org/licenses/
*
* For more information contact:
* OpenNMS(R) Licensing <license@opennms.org>
* http://www.opennms.org/
* http://www.opennms.com/
*******************************************************************************/
package org.opennms.netmgt.provision.support;
import java.io.IOException;
import java.io.InterruptedIOException;
import java.net.ConnectException;
import java.net.InetAddress;
import java.net.NoRouteToHostException;
import org.opennms.core.utils.InetAddressUtils;
import org.opennms.core.utils.LogUtils;
/**
* <p>Abstract BasicDetector class.</p>
*
* @author <a href=mailto:desloge@opennms.com>Donald Desloge</a>
* @version $Id: $
*/
public abstract class BasicDetector<Request, Response> extends SyncAbstractDetector {
private ClientConversation<Request, Response> m_conversation = new ClientConversation<Request, Response>();
/**
* <p>Constructor for BasicDetector.</p>
*
* @param serviceName a {@link java.lang.String} object.
* @param port a int.
* @param timeout a int.
* @param retries a int.
* @param <Request> a Request object.
* @param <Response> a Response object.
*/
protected BasicDetector(String serviceName, int port, int timeout, int retries) {
super(serviceName, port, timeout, retries);
}
/**
* <p>Constructor for BasicDetector.</p>
*
* @param serviceName a {@link java.lang.String} object.
* @param port a int.
*/
protected BasicDetector(String serviceName, int port) {
super(serviceName, port);
}
/** {@inheritDoc} */
@Override
public final boolean isServiceDetected(final InetAddress address) {
final String ipAddr = InetAddressUtils.str(address);
final int port = getPort();
final int retries = getRetries();
final int timeout = getTimeout();
LogUtils.infof(this, "isServiceDetected: Checking address: %s for %s capability on port %s", ipAddr, getServiceName(), getPort());
final Client<Request, Response> client = getClient();
for (int attempts = 0; attempts <= retries; attempts++) {
try {
client.connect(address, port, timeout);
LogUtils.infof(this, "isServiceDetected: Attempting to connect to address: %s, port: %d, attempt: #%s", ipAddr, port, attempts);
if (attemptConversation(client)) {
return true;
}
} catch (ConnectException e) {
// Connection refused!! Continue to retry.
LogUtils.infof(this, e, "isServiceDetected: %s: Unable to connect to address: %s port %d, attempt #%s",getServiceName(), ipAddr, port, attempts);
} catch (NoRouteToHostException e) {
// No Route to host!!!
LogUtils.infof(this, e, "isServiceDetected: %s: No route to address %s was available", getServiceName(), ipAddr);
} catch (InterruptedIOException e) {
// Expected exception
LogUtils.infof(this, e, "isServiceDetected: %s: Did not connect to to address %s port %d within timeout: %d attempt: %d", getServiceName(), ipAddr, port, timeout, attempts);
} catch (IOException e) {
LogUtils.errorf(this, e, "isServiceDetected: %s: An unexpected I/O exception occured contacting address %s port %d",getServiceName(), ipAddr, port);
} catch (Throwable t) {
LogUtils.errorf(this, t, "isServiceDetected: %s: Unexpected error trying to detect %s on address %s port %d", getServiceName(), getServiceName(), ipAddr, port);
} finally {
client.close();
}
}
return false;
}
/**
* <p>dispose</p>
*/
@Override
public void dispose(){
// Do nothing by default
}
/**
* <p>getClient</p>
*
* @return a {@link org.opennms.netmgt.provision.support.Client} object.
*/
abstract protected Client<Request, Response> getClient();
private boolean attemptConversation(Client<Request, Response> client) throws IOException, Exception {
return getConversation().attemptConversation(client);
}
/**
* <p>expectBanner</p>
*
* @param bannerValidator a {@link org.opennms.netmgt.provision.support.ResponseValidator} object.
*/
protected final void expectBanner(ResponseValidator<Response> bannerValidator) {
getConversation().expectBanner(bannerValidator);
}
/**
* <p>send</p>
*
* @param requestBuilder a {@link org.opennms.netmgt.provision.support.RequestBuilder} object.
* @param responseValidator a {@link org.opennms.netmgt.provision.support.ResponseValidator} object.
*/
protected final void send(RequestBuilder<Request> requestBuilder, ResponseValidator<Response> responseValidator) {
getConversation().addExchange(requestBuilder, responseValidator);
}
/**
* <p>send</p>
*
* @param request a Request object.
* @param responseValidator a {@link org.opennms.netmgt.provision.support.ResponseValidator} object.
*/
protected void send(Request request, ResponseValidator<Response> responseValidator) {
getConversation().addExchange(request, responseValidator);
}
/**
* <p>getConversation</p>
*
* @return a {@link org.opennms.netmgt.provision.support.ClientConversation} object.
*/
protected final ClientConversation<Request, Response> getConversation() {
return m_conversation;
}
}