All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.github.microcks.minion.async.consumer.NATSMessageConsumptionTask Maven / Gradle / Ivy

/*
 * Licensed to Laurent Broudoux (the "Author") under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. Author 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 io.github.microcks.minion.async.consumer;

import io.github.microcks.domain.Header;
import io.github.microcks.minion.async.AsyncTestSpecification;

import io.nats.client.Connection;
import io.nats.client.Dispatcher;
import io.nats.client.Nats;
import io.nats.client.Options;
import io.nats.client.impl.Headers;
import org.jboss.logging.Logger;

import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

/**
 * An implementation of MessageConsumptionTask that consumes a topic on an NATS.
 * Endpoint URL should be specified using the following form: nats://{brokerhost[:port]}
 * @author laurent
 */
public class NATSMessageConsumptionTask implements MessageConsumptionTask {

   /**
    * Get a JBoss logging logger.
    */
   private final Logger logger = Logger.getLogger(getClass());

   /**
    * The string for Regular Expression that helps validating acceptable
    * endpoints.
    */
   public static final String ENDPOINT_PATTERN_STRING = "nats://(?[^:]+(:\\d+)?)/(?.+)";

   /**
    * The Pattern for matching groups within the endpoint regular expression.
    */
   public static final Pattern ENDPOINT_PATTERN = Pattern.compile(ENDPOINT_PATTERN_STRING);

   private AsyncTestSpecification specification;

   private Connection subscriber;

   private String endpointTopic;

   private String options;

   /**
    * Create a new consumption task from an Async test specification.
    * @param testSpecification The specification holding endpointURL and timeout.
    */
   public NATSMessageConsumptionTask(AsyncTestSpecification testSpecification) {
      this.specification = testSpecification;
   }

   /**
    * Convenient static method for checking if this implementation will accept endpoint.
    * @param endpointUrl The endpoint URL to validate
    * @return True if endpointUrl can be used for connecting and consuming on
    * endpoint
    */
   public static boolean acceptEndpoint(String endpointUrl) {
      return endpointUrl != null && endpointUrl.matches(ENDPOINT_PATTERN_STRING);
   }

   @Override
   public List call() throws Exception {
      if (subscriber == null) {
         intializeNATSClient();
      }
      List messages = new ArrayList<>();

      // Start subscribing to the broker endpoint topic.
      Dispatcher d = subscriber.createDispatcher((msg) -> {
         String msgString = new String(msg.getData(), StandardCharsets.UTF_8);
         logger.info("Received a new NATS Message: " + msgString);
         // Build a ConsumedMessage from NATS message.
         ConsumedMessage message = new ConsumedMessage();
         message.setReceivedAt(System.currentTimeMillis());
         message.setHeaders(buildHeaders(msg.getHeaders()));
         message.setPayload(msg.getData());
         messages.add(message);
      });

      d.subscribe(endpointTopic);

      Thread.sleep(specification.getTimeoutMS());

      // Disconnect the subscriber before returning results.
      subscriber.close();
      return messages;
   }

   /** Build set of Microcks headers from NATS headers. */
   private Set
buildHeaders(Headers headers) { if (headers == null || headers.isEmpty()) { return null; } Set
results = new HashSet<>(); for (Map.Entry> entry : headers.entrySet()) { Header result = new Header(); result.setName(entry.getKey()); result.setValues(new HashSet<>(entry.getValue())); results.add(result); } return results; } /** * Close the resources used by this task. Namely the NATS subscriber and the option truststore. * @throws IOException should not happen. */ @Override public void close() throws IOException { if (subscriber != null) { try { subscriber.close(); } catch (InterruptedException e) { logger.warn("Closing NATS subscriber raised an exception", e); } } } private void intializeNATSClient() throws Exception { Matcher matcher = ENDPOINT_PATTERN.matcher(specification.getEndpointUrl().trim()); // Call matcher.find() to be able to use named expressions. matcher.find(); String endpointBrokerUrl = matcher.group("brokerUrl"); endpointTopic = matcher.group("topic"); Options.Builder optionsBuilder = new Options.Builder() .server(endpointBrokerUrl) .maxReconnects(10); if (specification.getSecret() != null) { if (specification.getSecret().getUsername() != null && specification.getSecret().getPassword() != null) { logger.debug("Adding username/password authentication from secret " + specification.getSecret().getName()); optionsBuilder.userInfo(specification.getSecret().getUsername(), specification.getSecret().getPassword()); } else if (specification.getSecret().getToken() != null) { logger.debug("Adding token authentication from secret " + specification.getSecret().getName()); optionsBuilder.token(specification.getSecret().getToken().toCharArray()); } } subscriber = Nats.connect(optionsBuilder.build()); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy