JmsPublisher.java
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.cxf.wsn.jms;
import java.io.StringWriter;
import java.util.HashMap;
import java.util.Map;
import java.util.logging.Level;
import java.util.logging.Logger;
import jakarta.jms.Connection;
import jakarta.jms.Destination;
import jakarta.jms.JMSException;
import jakarta.jms.Message;
import jakarta.jms.MessageConsumer;
import jakarta.jms.MessageProducer;
import jakarta.jms.Session;
import jakarta.jms.Topic;
import jakarta.xml.bind.JAXBContext;
import jakarta.xml.bind.JAXBException;
import org.apache.activemq.artemis.jms.client.ActiveMQTopic;
import org.apache.cxf.common.logging.LogUtils;
import org.apache.cxf.wsn.AbstractPublisher;
import org.oasis_open.docs.wsn.b_2.InvalidTopicExpressionFaultType;
import org.oasis_open.docs.wsn.b_2.NotificationMessageHolderType;
import org.oasis_open.docs.wsn.b_2.Notify;
import org.oasis_open.docs.wsn.b_2.TopicExpressionType;
import org.oasis_open.docs.wsn.br_2.PublisherRegistrationFailedFaultType;
import org.oasis_open.docs.wsn.br_2.RegisterPublisher;
import org.oasis_open.docs.wsn.br_2.ResourceNotDestroyedFaultType;
import org.oasis_open.docs.wsn.brw_2.PublisherRegistrationFailedFault;
import org.oasis_open.docs.wsn.brw_2.PublisherRegistrationRejectedFault;
import org.oasis_open.docs.wsn.brw_2.ResourceNotDestroyedFault;
import org.oasis_open.docs.wsn.bw_2.InvalidTopicExpressionFault;
import org.oasis_open.docs.wsn.bw_2.TopicNotSupportedFault;
import org.oasis_open.docs.wsrf.rw_2.ResourceUnknownFault;
public abstract class JmsPublisher extends AbstractPublisher {
private static final Logger LOGGER = LogUtils.getL7dLogger(JmsPublisher.class);
private Connection connection;
private JmsTopicExpressionConverter topicConverter;
private JAXBContext jaxbContext;
private Session advisory;
private MessageConsumer notificationConsumer;
private Map<Destination, Object> producers;
private final String notificationTopicName;
public JmsPublisher(String name) {
this(name, "activemq.notifications");
}
public JmsPublisher(String name, String notificationTopicName) {
super(name);
topicConverter = new JmsTopicExpressionConverter();
this.notificationTopicName = notificationTopicName;
try {
jaxbContext = JAXBContext.newInstance(Notify.class);
} catch (JAXBException e) {
throw new RuntimeException("Unable to create JAXB context", e);
}
}
public Connection getConnection() {
return connection;
}
public void setConnection(Connection connection) {
this.connection = connection;
}
@Override
public void notify(NotificationMessageHolderType messageHolder) {
Session session = null;
try {
Topic topic = topicConverter.toActiveMQTopic(messageHolder.getTopic());
session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
MessageProducer producer = session.createProducer(topic);
Notify notify = new Notify();
notify.getNotificationMessage().add(messageHolder);
StringWriter writer = new StringWriter();
jaxbContext.createMarshaller().marshal(notify, writer);
Message message = session.createTextMessage(writer.toString());
producer.send(message);
} catch (JMSException | JAXBException | InvalidTopicException e) {
LOGGER.log(Level.WARNING, "Error dispatching message", e);
} finally {
if (session != null) {
try {
session.close();
} catch (JMSException e) {
LOGGER.log(Level.FINE, "Error closing session", e);
}
}
}
}
@Override
protected void validatePublisher(RegisterPublisher registerPublisherRequest)
throws InvalidTopicExpressionFault, PublisherRegistrationFailedFault,
PublisherRegistrationRejectedFault, ResourceUnknownFault,
TopicNotSupportedFault {
super.validatePublisher(registerPublisherRequest);
try {
if (topic != null && !topic.isEmpty()) {
int size = topic.size();
ActiveMQTopic[] childrenDestinations = new ActiveMQTopic[size];
for (int i = 0; i < size; i++) {
childrenDestinations[i] = topicConverter.toActiveMQTopic(topic.get(i));
}
}
} catch (InvalidTopicException e) {
InvalidTopicExpressionFaultType fault = new InvalidTopicExpressionFaultType();
throw new InvalidTopicExpressionFault(e.getMessage(), fault);
}
}
@Override
protected void start() throws PublisherRegistrationFailedFault {
if (demand) {
try {
producers = new HashMap<>();
advisory = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
final Topic notificationsTopic = advisory.createTopic(notificationTopicName);
notificationConsumer = advisory.createConsumer(notificationsTopic);
notificationConsumer.setMessageListener(this::onMessage);
} catch (Exception e) {
PublisherRegistrationFailedFaultType fault = new PublisherRegistrationFailedFaultType();
throw new PublisherRegistrationFailedFault("Error starting demand-based publisher", fault, e);
}
}
}
protected void destroy() throws ResourceNotDestroyedFault {
try {
if (notificationConsumer != null) {
notificationConsumer.close();
}
if (advisory != null) {
advisory.close();
}
} catch (Exception e) {
ResourceNotDestroyedFaultType fault = new ResourceNotDestroyedFaultType();
throw new ResourceNotDestroyedFault("Error destroying publisher", fault, e);
} finally {
super.destroy();
}
}
public synchronized void onMessage(Message event) {
try {
// See please https://activemq.apache.org/components/artemis/documentation/latest/management.html
final String type = event.getStringProperty("_AMQ_NotifType");
final String routing = event.getStringProperty("_AMQ_RoutingName");
if (routing != null) {
final TopicExpressionType topic = topicConverter.toTopicExpression(routing);
final Destination destination = topicConverter.toActiveMQTopic(topic);
Object producer = producers.get(destination);
if ("CONSUMER_CREATED".equalsIgnoreCase(type)) {
final int consumerCount = event.getIntProperty("_AMQ_ConsumerCount");
if (consumerCount > 0 && producer == null) {
// start subscription
producer = startSubscription(topic);
producers.put(destination, producer);
}
} else if ("CONSUMER_CLOSED".equalsIgnoreCase(type)) {
final int consumerCount = event.getIntProperty("_AMQ_ConsumerCount");
if (consumerCount == 0 && producer != null) {
Object sub = producers.remove(destination);
// destroy subscription
stopSubscription(sub);
}
}
}
} catch (JMSException | InvalidTopicException ex) {
LOGGER.log(Level.WARNING, "Error consuming message", ex);
}
}
protected abstract Object startSubscription(TopicExpressionType topic);
protected abstract void stopSubscription(Object sub);
}