/*******************************************************************************
* 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);
}
}