2 * Copyright (c) 2015 Cisco Systems, Inc. and others. All rights reserved.
4 * This program and the accompanying materials are made available under the
5 * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6 * and is available at http://www.eclipse.org/legal/epl-v10.html
8 package org.opendaylight.controller.messagebus.eventsources.netconf;
11 import java.util.ArrayList;
12 import java.util.concurrent.ConcurrentHashMap;
14 import javax.xml.parsers.DocumentBuilder;
15 import javax.xml.parsers.DocumentBuilderFactory;
16 import javax.xml.parsers.ParserConfigurationException;
17 import javax.xml.transform.dom.DOMSource;
19 import org.opendaylight.controller.md.sal.dom.api.DOMNotification;
20 import org.opendaylight.controller.md.sal.dom.api.DOMNotificationListener;
21 import org.opendaylight.yang.gen.v1.urn.cisco.params.xml.ns.yang.messagebus.eventaggregator.rev141202.TopicId;
22 import org.opendaylight.yang.gen.v1.urn.cisco.params.xml.ns.yang.messagebus.eventsource.rev141202.EventSourceStatus;
23 import org.opendaylight.yang.gen.v1.urn.cisco.params.xml.ns.yang.messagebus.eventsource.rev141202.EventSourceStatusNotification;
24 import org.opendaylight.yang.gen.v1.urn.cisco.params.xml.ns.yang.messagebus.eventsource.rev141202.EventSourceStatusNotificationBuilder;
25 import org.opendaylight.yangtools.yang.common.QName;
26 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier;
27 import org.opendaylight.yangtools.yang.data.api.schema.AnyXmlNode;
28 import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode;
29 import org.opendaylight.yangtools.yang.data.impl.schema.Builders;
30 import org.opendaylight.yangtools.yang.model.api.SchemaPath;
31 import org.slf4j.Logger;
32 import org.slf4j.LoggerFactory;
33 import org.w3c.dom.Document;
34 import org.w3c.dom.Element;
36 import com.google.common.base.Optional;
37 import com.google.common.base.Preconditions;
39 public class ConnectionNotificationTopicRegistration extends NotificationTopicRegistration {
41 private static final Logger LOG = LoggerFactory.getLogger(ConnectionNotificationTopicRegistration.class);
43 public static final SchemaPath EVENT_SOURCE_STATUS_PATH = SchemaPath.create(true, QName.create(EventSourceStatusNotification.QNAME, "event-source-status"));
44 private static final NodeIdentifier EVENT_SOURCE_STATUS_ARG = new NodeIdentifier(EventSourceStatusNotification.QNAME);
45 private static final String XMLNS_ATTRIBUTE_KEY = "xmlns";
46 private static final String XMLNS_URI = "http://www.w3.org/2000/xmlns/";
48 private final DOMNotificationListener domNotificationListener;
49 private ConcurrentHashMap<SchemaPath, ArrayList<TopicId>> notificationTopicMap = new ConcurrentHashMap<>();
51 public ConnectionNotificationTopicRegistration(String SourceName, DOMNotificationListener domNotificationListener) {
52 super(NotificationSourceType.ConnectionStatusChange, SourceName, EVENT_SOURCE_STATUS_PATH.getLastComponent().getNamespace().toString());
53 this.domNotificationListener = Preconditions.checkNotNull(domNotificationListener);
54 LOG.info("Connection notification source has been initialized...");
56 setReplaySupported(false);
60 public void close() throws Exception {
61 LOG.info("Connection notification - publish Deactive");
62 publishNotification(EventSourceStatus.Deactive);
63 notificationTopicMap.clear();
68 void activateNotificationSource() {
69 LOG.info("Connection notification - publish Active");
70 publishNotification(EventSourceStatus.Active);
74 void deActivateNotificationSource() {
75 LOG.info("Connection notification - publish Inactive");
76 publishNotification(EventSourceStatus.Inactive);
80 void reActivateNotificationSource() {
81 LOG.info("Connection notification - reactivate - publish active");
82 publishNotification(EventSourceStatus.Active);
86 boolean registerNotificationTopic(SchemaPath notificationPath, TopicId topicId) {
87 if(validateNotifactionSchemaPath(notificationPath) == false){
88 LOG.debug("Bad SchemaPath for notification try to register");
91 ArrayList<TopicId> topicIds = getNotificationTopicIds(notificationPath);
93 topicIds = new ArrayList<>();
94 topicIds.add(topicId);
96 if(topicIds.contains(topicId) == false){
97 topicIds.add(topicId);
100 notificationTopicMap.put(notificationPath, topicIds);
105 ArrayList<TopicId> getNotificationTopicIds(SchemaPath notificationPath) {
106 return notificationTopicMap.get(notificationPath);
110 void unRegisterNotificationTopic(TopicId topicId) {
111 // TODO: need code when EventAggregator.destroyTopic will be implemented
114 private boolean validateNotifactionSchemaPath(SchemaPath notificationPath){
115 if(notificationPath == null){
118 URI notificationNameSpace = notificationPath.getLastComponent().getNamespace();
119 return getNotificationUrnPrefix().startsWith(notificationNameSpace.toString());
122 private void publishNotification(EventSourceStatus eventSourceStatus){
124 final EventSourceStatusNotification notification = new EventSourceStatusNotificationBuilder()
125 .setStatus(eventSourceStatus)
127 domNotificationListener.onNotification(createNotification(notification));
130 private DOMNotification createNotification(EventSourceStatusNotification notification){
131 final ContainerNode cn = Builders.containerBuilder()
132 .withNodeIdentifier(EVENT_SOURCE_STATUS_ARG)
133 .withChild(encapsulate(notification))
135 DOMNotification dn = new DOMNotification() {
138 public SchemaPath getType() {
139 return EVENT_SOURCE_STATUS_PATH;
143 public ContainerNode getBody() {
150 private AnyXmlNode encapsulate(EventSourceStatusNotification notification){
152 DocumentBuilderFactory docFactory = DocumentBuilderFactory.newInstance();
153 DocumentBuilder docBuilder;
156 docBuilder = docFactory.newDocumentBuilder();
157 } catch (ParserConfigurationException e) {
158 throw new IllegalStateException("Can not create XML DocumentBuilder");
161 Document doc = docBuilder.newDocument();
163 final Optional<String> namespace = Optional.of(EVENT_SOURCE_STATUS_ARG.getNodeType().getNamespace().toString());
164 final Element rootElement = createElement(doc , "EventSourceStatusNotification", namespace);
166 final Element sourceElement = doc.createElement("status");
167 sourceElement.appendChild(doc.createTextNode(notification.getStatus().name()));
168 rootElement.appendChild(sourceElement);
171 return Builders.anyXmlBuilder().withNodeIdentifier(EVENT_SOURCE_STATUS_ARG)
172 .withValue(new DOMSource(rootElement))
177 // Helper to create root XML element with correct namespace and attribute
178 private Element createElement(final Document document, final String qName, final Optional<String> namespaceURI) {
179 if(namespaceURI.isPresent()) {
180 final Element element = document.createElementNS(namespaceURI.get(), qName);
181 String name = XMLNS_ATTRIBUTE_KEY;
182 if(element.getPrefix() != null) {
183 name += ":" + element.getPrefix();
185 element.setAttributeNS(XMLNS_URI, name, namespaceURI.get());
188 return document.createElement(qName);