/******************************************************************************* * This file is part of OpenNMS(R). * * Copyright (C) 2007-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.poller.monitors; import java.io.IOException; import java.net.InetAddress; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.LinkedHashMap; import java.util.List; import java.util.Map; import org.opennms.core.utils.CollectionMath; import org.opennms.core.utils.ParameterMap; import org.opennms.core.utils.ThreadCategory; import org.opennms.netmgt.icmp.PingConstants; import org.opennms.netmgt.icmp.PingerFactory; import org.opennms.netmgt.model.PollStatus; import org.opennms.netmgt.poller.Distributable; import org.opennms.netmgt.poller.MonitoredService; import org.opennms.netmgt.poller.NetworkInterface; import org.opennms.netmgt.poller.NetworkInterfaceNotSupportedException; /** * <P> * This class is designed to be used by the service poller framework to test the * availability of the ICMP service on remote interfaces. The class implements * the ServiceMonitor interface that allows it to be used along with other * plug-ins by the service poller framework. * </P> * * @author <A HREF="mailto:tarus@opennms.org">Tarus Balog</A> * @author <A HREF="mailto:ranger@opennms.org">Benjamin Reed</A> * @author <A HREF="http://www.opennms.org/">OpenNMS</A> */ @Distributable final public class StrafePingMonitor extends AbstractServiceMonitor { private static final int DEFAULT_MULTI_PING_COUNT = 20; private static final long DEFAULT_PING_INTERVAL = 50; private static final int DEFAULT_FAILURE_PING_COUNT = 20; /** * Constructs a new monitor. * * @throws java.io.IOException if any. */ public StrafePingMonitor() throws IOException { } /** * {@inheritDoc} * * <P> * Poll the specified address for ICMP service availability. * </P> * * <P> * The ICMP service monitor relies on Discovery for the actual generation of * IMCP 'ping' requests. A JSDT session with two channels (send/receive) is * utilized for passing poll requests and receiving poll replies from * discovery. All exchanges are SOAP/XML compliant. * </P> */ public PollStatus poll(MonitoredService svc, Map<String, Object> parameters) { NetworkInterface<InetAddress> iface = svc.getNetInterface(); // Get interface address from NetworkInterface // if (iface.getType() != NetworkInterface.TYPE_INET) throw new NetworkInterfaceNotSupportedException("Unsupported interface type, only TYPE_INET currently supported"); ThreadCategory log = ThreadCategory.getInstance(this.getClass()); PollStatus serviceStatus = PollStatus.unavailable(); InetAddress host = (InetAddress) iface.getAddress(); List<Number> responseTimes = null; try { // get parameters // long timeout = ParameterMap.getKeyedLong(parameters, "timeout", PingConstants.DEFAULT_TIMEOUT); int count = ParameterMap.getKeyedInteger(parameters, "ping-count", DEFAULT_MULTI_PING_COUNT); long pingInterval = ParameterMap.getKeyedLong(parameters, "wait-interval", DEFAULT_PING_INTERVAL); int failurePingCount = ParameterMap.getKeyedInteger(parameters, "failure-ping-count", DEFAULT_FAILURE_PING_COUNT); responseTimes = new ArrayList<Number>(PingerFactory.getInstance().parallelPing(host, count, timeout, pingInterval)); if (CollectionMath.countNull(responseTimes) >= failurePingCount) { if (log().isDebugEnabled()) { log().debug("Service " +svc.getSvcName()+ " on interface " +svc.getIpAddr()+ " is down, but continuing to gather latency data"); } serviceStatus = PollStatus.unavailable("the failure ping count (" + failurePingCount + ") was reached"); } else { serviceStatus = PollStatus.available(); } Collections.sort(responseTimes, new Comparator<Number>() { public int compare(Number arg0, Number arg1) { if (arg0 == null) { return -1; } else if (arg1 == null) { return 1; } else if (arg0.doubleValue() == arg1.doubleValue()) { return 0; } else { return arg0.doubleValue() < arg1.doubleValue() ? -1 : 1; } } }); Map<String, Number> returnval = new LinkedHashMap<String, Number>(); for (int i = 0; i < responseTimes.size(); i++) { returnval.put("ping" + (i + 1), responseTimes.get(i)); } returnval.put("loss", CollectionMath.countNull(responseTimes)); returnval.put("median", CollectionMath.median(responseTimes)); returnval.put("response-time", CollectionMath.average(responseTimes)); serviceStatus.setProperties(returnval); } catch (Throwable e) { log.debug("failed to ping " + host, e); } return serviceStatus; } }