/******************************************************************************* * This file is part of OpenNMS(R). * * Copyright (C) 2006-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.pollables; import java.net.InetAddress; import java.util.Date; import java.util.concurrent.Callable; import org.opennms.core.utils.InetAddressUtils; import org.opennms.netmgt.EventConstants; import org.opennms.netmgt.model.PollStatus; import org.opennms.netmgt.xml.event.Event; /** * Represents a PollableInterface * * @author <a href="mailto:brozow@opennms.org">Mathew Brozowski</a> * @version $Id: $ */ public class PollableInterface extends PollableContainer { private final InetAddress m_addr; /** * <p>Constructor for PollableInterface.</p> * * @param node a {@link org.opennms.netmgt.poller.pollables.PollableNode} object. * @param addr a {@link java.net.InetAddress} object. */ public PollableInterface(PollableNode node, InetAddress addr) { super(node, Scope.INTERFACE); m_addr = addr; } /** * <p>getNode</p> * * @return a {@link org.opennms.netmgt.poller.pollables.PollableNode} object. */ public PollableNode getNode() { return (PollableNode)getParent(); } private void setNode(PollableNode newNode) { setParent(newNode); } /** * <p>getNetwork</p> * * @return a {@link org.opennms.netmgt.poller.pollables.PollableNetwork} object. */ public PollableNetwork getNetwork() { return getNode().getNetwork(); } /** * <p>getContext</p> * * @return a {@link org.opennms.netmgt.poller.pollables.PollContext} object. */ public PollContext getContext() { return getNode().getContext(); } /** * <p>getIpAddr</p> * * @return a {@link java.lang.String} object. */ public String getIpAddr() { return InetAddressUtils.str(m_addr); } /** * <p>getAddress</p> * * @return a {@link java.net.InetAddress} object. */ public InetAddress getAddress() { return m_addr; } /** * <p>getNodeId</p> * * @return a int. */ public int getNodeId() { return getNode().getNodeId(); } /** * <p>getNodeLabel</p> * * @return a {@link java.lang.String} object. */ public String getNodeLabel() { return getNode().getNodeLabel(); } /** * <p>createService</p> * * @param svcName a {@link java.lang.String} object. * @return a {@link org.opennms.netmgt.poller.pollables.PollableService} object. */ public PollableService createService(final String svcName) { return withTreeLock(new Callable<PollableService>() { public PollableService call() { PollableService svc = new PollableService(PollableInterface.this, svcName); addMember(svc); return svc; } }); } /** * <p>getService</p> * * @param svcName a {@link java.lang.String} object. * @return a {@link org.opennms.netmgt.poller.pollables.PollableService} object. */ public PollableService getService(String svcName) { return (PollableService)getMember(svcName); } /** {@inheritDoc} */ protected Object createMemberKey(PollableElement member) { PollableService svc = (PollableService)member; return svc.getSvcName(); } /** {@inheritDoc} */ protected void visitThis(PollableVisitor v) { super.visitThis(v); v.visitInterface(this); } /** * <p>recalculateStatus</p> */ public void recalculateStatus() { PollableService criticalSvc = getCriticalService(); if (criticalSvc != null) { criticalSvc.recalculateStatus(); updateStatus(criticalSvc.getStatus().isUp() ? PollStatus.up() : PollStatus.down()); } else { super.recalculateStatus(); } } /** * @return */ private PollableService getCriticalService() { return getService(getContext().getCriticalServiceName()); } /** * <p>selectPollElement</p> * * @return a {@link org.opennms.netmgt.poller.pollables.PollableElement} object. */ public PollableElement selectPollElement() { PollableService critSvc = getCriticalService(); return (critSvc != null ? critSvc : super.selectPollElement()); } /** {@inheritDoc} */ protected PollStatus poll(PollableElement elem) { PollableService critSvc = getCriticalService(); if (getStatus().isUp() || critSvc == null || elem == critSvc) return super.poll(elem); return PollStatus.down(); } /** {@inheritDoc} */ public PollStatus pollRemainingMembers(PollableElement member) { PollableService critSvc = getCriticalService(); if (critSvc != null && getStatus().isUp()) { if (member != critSvc) critSvc.poll(); return critSvc.getStatus().isUp() ? PollStatus.up() : PollStatus.down(); } if (getContext().isPollingAllIfCritServiceUndefined()) return super.pollRemainingMembers(member); else { return getMemberStatus(); } } /** {@inheritDoc} */ public Event createDownEvent(Date date) { return getContext().createEvent(EventConstants.INTERFACE_DOWN_EVENT_UEI, getNodeId(), getAddress(), null, date, getStatus().getReason()); } /** {@inheritDoc} */ public Event createUpEvent(Date date) { return getContext().createEvent(EventConstants.INTERFACE_UP_EVENT_UEI, getNodeId(), getAddress(), null, date, getStatus().getReason()); } /** * <p>toString</p> * * @return a {@link java.lang.String} object. */ public String toString() { return getNode()+":"+getIpAddr(); } /** * <p>reparentTo</p> * * @param newNode a {@link org.opennms.netmgt.poller.pollables.PollableNode} object. */ public void reparentTo(final PollableNode newNode) { final PollableNode oldNode = getNode(); if (oldNode.equals(newNode)) return; // always lock the nodes in nodeId order so deadlock is not possible final PollableNode firstNode = (oldNode.getNodeId() <= newNode.getNodeId() ? oldNode : newNode); final PollableNode secondNode = (oldNode.getNodeId() <= newNode.getNodeId() ? newNode : oldNode); final Runnable reparent = new Runnable() { public void run() { oldNode.resetStatusChanged(); newNode.resetStatusChanged(); int oldNodeId = getNodeId(); String oldIp = getIpAddr(); int newNodeId = newNode.getNodeId(); oldNode.removeMember(PollableInterface.this); newNode.addMember(PollableInterface.this); setNode(newNode); getContext().reparentOutages(oldIp, oldNodeId, newNodeId); if (getCause() == null || getCause().equals(oldNode.getCause())) { // the current interface outage is a node outage or no outage at all if (newNode.getCause() != null) { // if the new Node has a node outage then we recursively set the // causes so when process events we properly handle the causes PollableVisitor visitor = new PollableVisitorAdaptor() { public void visitElement(PollableElement element) { boolean matches = (element.getCause() == null ? oldNode.getCause() == null : element.getCause().equals(oldNode.getCause())); if (matches) { element.setCause(newNode.getCause()); } } }; visit(visitor); } } // process the status changes related to the Date date = new Date(); oldNode.recalculateStatus(); oldNode.processStatusChange(date); newNode.recalculateStatus(); newNode.processStatusChange(date); } }; Runnable lockSecondNodeAndRun = new Runnable() { public void run() { secondNode.withTreeLock(reparent); } }; firstNode.withTreeLock(lockSecondNodeAndRun); } }