I need to mock a RabbitMQ in my unit Test

24,582

Solution 1

As I understand it, there are two things trying to be tested in the question:

  • TLS configuration to connect to RabbitMQ
  • basicPublish / basicConsume (what's called delivery) behavior regarding interactions with the rest of the application

For the first one, as TLS itself is being tested, only connecting to a real instance of RabbitMQ with correct truststore configured will prove that configuration is working

For the second one however, for tests demonstrating features of the app (with tools like Cucumber for readability), you may try a library i'm working on: rabbitmq-mock (and that's why I'm digging up an old post)

Just include it as dependency:

<dependency>
    <groupId>com.github.fridujo</groupId>
    <artifactId>rabbitmq-mock</artifactId>
    <version>1.0.14</version>
    <scope>test</scope>
</dependency>

And replace new ConnectionFactory() by new MockConnectionFactory() in your unit test.

Samples are available in the project: https://github.com/fridujo/rabbitmq-mock/blob/master/src/test/java/com/github/fridujo/rabbitmq/mock/IntegrationTest.java

Solution 2

I know, it is an old question, still as there is no answer so far. What helped me a lot at the same question, is the following blog post: https://tamasgyorfi.net/2016/04/21/writing-integration-tests-for-rabbitmq-based-components/. It uses Apache QPID (not ActiveMQ as suggested in the OP) and it has support for AMQP 0.9.1.

Share:
24,582

Related videos on Youtube

Miguel
Author by

Miguel

Updated on February 10, 2020

Comments

  • Miguel
    Miguel over 4 years

    I am using a RabbitMQ in my project.

    I have in my consumer the code of the client part of rabbitMQ and the connection need a tls1.1 to connect with the real MQ.

    I want to test this code in my JUnit test and to mock the message delivery to my consumer.

    I see in google several examples with different tools how camel rabbit or activeMQ but this tools works with amqp 1.0 and rabbitMQ only works in amqp 0.9 .

    Someone had this problem?

    Thanks!

    UPDATE

    This is the code to testing to receive a json from the queue.

    package com.foo.foo.queue;
    
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.IOException;
    import java.net.URL;
    import java.security.*;
    import java.security.cert.CertificateException;
    import javax.net.ssl.*;
    
    import org.apache.commons.lang3.StringUtils;
    import org.apache.log4j.LogManager;
    import org.apache.log4j.Logger;
    import org.json.JSONObject;
    
    import com.foo.foo.Constants.Constants;
    import com.foo.foo.core.ConfigurationContainer;
    import com.foo.foo.policyfinders.PolicyFinder;
    import com.rabbitmq.client.Channel;
    import com.rabbitmq.client.Connection;
    import com.rabbitmq.client.ConnectionFactory;
    import com.rabbitmq.client.QueueingConsumer;
    
    public class BrokerThreadHLConsumer extends Thread {
    
    private static BrokerThreadHLConsumer instance;
    
    private static final Logger log = LogManager.getLogger(BrokerThreadHLConsumer.class);
    
    private Channel channel;
    private String queueName;
    private PolicyFinder PolicyFinder;
    private Connection connection;
    private QueueingConsumer consumer;
    
    private boolean loop;
    
    private BrokerThreadHLConsumer() throws IOException {
        ConnectionFactory factory = new ConnectionFactory();
        char[] keyPassphrase = "clientrabbit".toCharArray();
        KeyStore keyStoreCacerts;
        ConfigurationContainer configurationContainer = ConfigurationContainer.getInstance();
        String exchangeName = configurationContainer.getProperty(Constants.EXCHANGE_NAME);
        String rabbitHost = configurationContainer.getProperty(Constants.RABBITMQ_SERVER_HOST_VALUE);
        try {
            /* Public key cacerts to connect to message queue*/
            keyStoreCacerts = KeyStore.getInstance("PKCS12");
            URL resourcePublicKey = this.getClass().getClassLoader().getResource("certs/client.keycert.p12");
            File filePublicKey = new File(resourcePublicKey.toURI());
            keyStoreCacerts.load(new FileInputStream(filePublicKey), keyPassphrase);
            KeyManagerFactory keyManager;
    
            keyManager = KeyManagerFactory.getInstance("SunX509");
            keyManager.init(keyStoreCacerts, keyPassphrase);
    
            char[] trustPassphrase = "changeit".toCharArray();
            KeyStore tks;
    
            tks = KeyStore.getInstance("JCEKS");
    
            URL resourceCacerts = this.getClass().getClassLoader().getResource("certs/cacerts");
            File fileCacerts = new File(resourceCacerts.toURI());
    
            tks.load(new FileInputStream(fileCacerts), trustPassphrase);
    
            TrustManagerFactory tmf;
            tmf = TrustManagerFactory.getInstance("SunX509");
            tmf.init(tks);
    
            SSLContext c = SSLContext.getInstance("TLSv1.1");
            c.init(keyManager.getKeyManagers(), tmf.getTrustManagers(), null);
    
            factory.setUri(rabbitHost);
            factory.useSslProtocol(c);
            connection = factory.newConnection();
            channel = connection.createChannel();
            channel.exchangeDeclare(exchangeName, "fanout");
            queueName = channel.queueDeclare().getQueue();
            channel.queueBind(queueName, exchangeName, "");
    
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        } catch (CertificateException e) {
            e.printStackTrace();
        } catch (KeyStoreException e) {
            e.printStackTrace();
        } catch (UnrecoverableKeyException e) {
            e.printStackTrace();
        } catch (KeyManagementException e1) {
            e1.printStackTrace();
        } catch (Exception e) {
            log.error("Couldn't instantiate a channel with the broker installed in " + rabbitHost);
            log.error(e.getStackTrace());
            e.printStackTrace();
        }
    }
    
    public static BrokerThreadHLConsumer getInstance() throws CertificateException, UnrecoverableKeyException, NoSuchAlgorithmException, KeyStoreException, KeyManagementException, IOException {
        if (instance == null)
            instance = new BrokerThreadHLConsumer();
        return instance;
    }
    
    public void run() {
        if (PolicyFinder != null) {
            try {
                consumer = new QueueingConsumer(channel);
                channel.basicConsume(queueName, true, consumer);
                log.info("Consumer broker started and waiting for messages");
                loop = true;
                while (loop) {
                    try {
                        QueueingConsumer.Delivery delivery = consumer.nextDelivery();
                        String message = new String(delivery.getBody());
                        JSONObject obj = new JSONObject(message);
                        log.info("Message received from broker " + obj);
                        if (StringUtils.isNotEmpty(message) && !PolicyFinder.managePolicySet(obj)) {
                            log.error("PolicySet error: error upgrading the policySet");
                        }
                    } catch (Exception e) {
                        log.error("Receiving message error");
                        log.error(e);
                    }
                }
            } catch (IOException e) {
                log.error("Consumer couldn't start");
                log.error(e.getStackTrace());
            }
        } else {
            log.error("Consumer couldn't start cause of PolicyFinder is null");
        }
    }
    
    public void close() {
        loop = false;
        try {
            consumer.getChannel().basicCancel(consumer.getConsumerTag());
        } catch (IOException e) {
            e.printStackTrace();
        }
        try {
            channel.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
        try {
            connection.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
    
    public void setLuxPolicyFinder(PolicyFinder PolicyFinder) {
        this.PolicyFinder = PolicyFinder;
    }
    }
    
    • Bilbo Baggins
      Bilbo Baggins about 8 years
      Code to be tested please? what have you tried so far?
    • Augusto
      Augusto about 8 years
      There's no much point in unit testing an integration point, as it doesn't provide any value. After all, you can mock the whole API in ways that don't work with the real broker. I'm not saying you shouldn't test this, but rather use an integration test. Have a test instance of rabbitMQ running in your infrastructure and run an integration test against it. You can randomly create a channel for each time the test runs so there's no conflict. Alternatively, if you're using docker or something similar, you can spin up a private broker.
    • NoobEditor
      NoobEditor over 6 years
      sometimes SO behaves like humanity when it comes to answers - hopeless!!!