Использование Log4J JMSAppender с ActiveMQ

Я пытаюсь создать доказательство ведения журнала доставки концепции из Log4J через JMS с использованием log4J JMSAppender. Я пробовал ActiveMQ и прилагаемый к нему пример. Я разобрал этот пример на части и сделал его более общим и совместимым с несколькими платформами.

Похоже, у меня все в порядке, так как я вижу, что происходит соединение с ActiveMQ, но код зависает, когда я получаю InitialContext (с установленным -Dlog4j.debug клиентские классы ActiveMQ, похоже, вызывают log4J и загружают свойства, которые, в свою очередь, пытаются для подключения к JMS для JMSAppender), а затем код просто зависает. Я попытался изолировать сообщения журнала, направляемые в JMS, определив приложение только для одного именованного регистратора, а пакет org.apache.activemq настроен на использование ConsoleAppender

Тот же код отлично работает, когда он указывает на сервер Weblogic с настроенной очередью JMS, но для максимальной совместимости мне нужно попытаться заставить его работать с ActiveMQ.

Есть ли какой-то «волшебный» бит конфигурации, которого мне не хватает, чтобы ActiveMQ работал правильно?

-- несколько примеров из проделанной работы, чтобы немного конкретизировать этот вопрос, теперь у меня есть код

log4j-jms.properties

log4j.rootLogger=INFO, stdout

## Be sure that ActiveMQ messages are not logged to 'jms' appender

log4j.logger.org.apache=ERROR, stdout
log4j.logger.javax=ERROR,stdout
log4j.logger.java=ERROR,stdout

log4j.logger.demo=DEBUG,jms

log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%d %-5p %c - %m%n

## Configure 'jms' appender. You'll also need jndi.properties file in order to make it work
log4j.appender.jms=org.apache.log4j.net.JMSAppender
log4j.appender.jms.InitialContextFactoryName=org.apache.activemq.jndi.ActiveMQInitialContextFactory
log4j.appender.jms.ProviderURL=tcp://localhost:61616
log4j.appender.jms.TopicBindingName=topic.logTopic
log4j.appender.jms.TopicConnectionFactoryBindingName=ConnectionFactory

Цель этого состояла в том, чтобы создать именованную демонстрацию приложения и в образце кода захватить это для регистрации, чтобы убедиться, что ведение журнала activemq не пытается отправить себя в JMS.

пример кода. Это немного беспорядок, так как я пытался заставить его работать. В его нынешнем виде он будет работать, когда я укажу его на Weblogic и аналогичным образом переключу конфигурацию log4j. Цель этого кода заключалась в том, чтобы убедиться, что слушатель темы работает в отдельном потоке.

NewLog4jJMSAppenderExample.java

import javax.jms.*;
import javax.naming.Context;
import javax.naming.InitialContext;
import javax.naming.NamingException;
import org.apache.log4j.Logger;
import org.apache.log4j.spi.LoggingEvent;

import java.util.Properties;
/**
 * A simple example of log4j jms appender in conjuction with ActiveMQ
 */
public class NewLog4jJMSAppenderExample {
    Runnable listener;
    Thread runner;

    private enum MQImplementation {
        ActiveMQ, Weblogic
    };

    public NewLog4jJMSAppenderExample() {
        // create a logTopic topic consumer

        listener = new BigEars();
        System.out.println("******* Listener Created **********");

        runner = new Thread(listener);
        runner.start();

        try {
            Thread.sleep(10);
        } catch (InterruptedException e) {
            e.printStackTrace();  //To change body of catch statement use File | Settings | File Templates.
        }

    }

    public static void main(String[] args) throws Exception {
        System.out.println("******* I HAVE STARTED **********");


        new NewLog4jJMSAppenderExample();


        System.out.println("******* LOGGING **********");

        // log a message

        Logger log = Logger.getLogger("demo");
        log.error("Test log");

        Thread.sleep(100000);

        System.exit(1);

    }

    public class BigEars implements Runnable, MessageListener {
        ConnectionFactory factory;
        Connection conn;
        Session sess;
        MessageConsumer consumer;

        public BigEars() {

            MQImplementation inUse = MQImplementation.ActiveMQ;

            System.out.println("Constructing Bigears");
            try {
                Properties env = new Properties();

                switch (inUse) {

                    case Weblogic:
                        env.put(Context.INITIAL_CONTEXT_FACTORY,
                                "weblogic.jndi.WLInitialContextFactory");
                        env.put(Context.PROVIDER_URL,
                                "t3://localhost:7001");
                        break;

                    case ActiveMQ:
                        env.put(Context.INITIAL_CONTEXT_FACTORY,
                                "org.apache.activemq.jndi.ActiveMQInitialContextFactory");
                        env.put(Context.PROVIDER_URL,
                                "tcp://localhost:61616");
                        break;
                }

                System.out.println("Initial Context");

                InitialContext jndi = new InitialContext(env);
                System.out.println("Factory");
                factory = (TopicConnectionFactory) jndi.lookup("ConnectionFactory");

                Topic theTopic = (Topic) jndi.lookup("topic.logTopic");


                System.out.println("Connection");
                conn = factory.createConnection();

                System.out.println("******* I HAVE set up and created connection **********");
                System.out.println("session");
                sess = conn.createSession(false, Session.AUTO_ACKNOWLEDGE);
                System.out.println("consumer");
                consumer = sess.createConsumer(theTopic);
                System.out.println("listener");
                consumer.setMessageListener(this);

                conn.start();



            } catch (JMSException jme) {
                System.out.println(jme);
            } catch (NamingException ne) {
                System.out.println(ne);
            }
        }


        public void run() {
            try {
                System.out.println("******* zzzzzzzz! **********");

                Thread.sleep(100000);
            } catch (Exception e) {
                e.printStackTrace();  //To change body of catch statement use File | Settings | File Templates.
            }
        }

        public void onMessage(Message message) {
            try {
                try {
                    System.out.println("******* I GOT A MESSAGE **********");
                    // receive log event in your consumer
                    System.out.println(message.toString());

                    LoggingEvent event = (LoggingEvent) (((ObjectMessage) message).getObject());
                    System.out.println("Received log [" + event.getLevel() + "]: " + event.getMessage());
                } catch (Exception e) {
                    e.printStackTrace();
                }

            } finally {

                try {
                    consumer.close();
                    sess.close();
                    conn.close();
                } catch (JMSException jme) {
                    System.out.println(jme);
                }
            }
        }

} }

логирование отображается, когда установлен параметр -Dlog4j.debug

******* I HAVE STARTED **********
Constructing Bigears
Initial Context
log4j: Trying to find [log4j-jms.properties] using context classloader sun.misc.Launcher$AppClassLoader@2c2bbd86.
log4j: Using URL [file:/Users/kevin/Desktop/apache-activemq-5.3.0/example/target/classes/log4j-jms.properties] for automatic log4j configuration.
log4j: Reading configuration from URL file:/Users/kevin/Desktop/apache-activemq-5.3.0/example/target/classes/log4j-jms.properties
log4j: Parsing for [root] with value=[INFO, stdout].
log4j: Level token is [INFO].
log4j: Category root set to INFO
log4j: Parsing appender named "stdout".
log4j: Parsing layout options for "stdout".
log4j: Setting property [conversionPattern] to [%d %-5p %c - %m%n].
log4j: End of parsing for "stdout".
log4j: Parsed "stdout" options.
log4j: Parsing for [org.apache] with value=[ERROR, stdout].
log4j: Level token is [ERROR].
log4j: Category org.apache set to ERROR
log4j: Parsing appender named "stdout".
log4j: Appender "stdout" was already parsed.
log4j: Handling log4j.additivity.org.apache=[null]
log4j: Parsing for [demo] with value=[DEBUG,jms].
log4j: Level token is [DEBUG].
log4j: Category demo set to DEBUG
log4j: Parsing appender named "jms".
log4j: Setting property [initialContextFactoryName] to [org.apache.activemq.jndi.ActiveMQInitialContextFactory].
log4j: Setting property [topicBindingName] to [topic.logTopic].
log4j: Setting property [topicConnectionFactoryBindingName] to [ConnectionFactory].
log4j: Setting property [providerURL] to [tcp://localhost:61616].
log4j: Getting initial context.
log4j: Looking up [ConnectionFactory]
log4j: About to create TopicConnection.
log4j: Creating TopicSession, non-transactional, in AUTO_ACKNOWLEDGE mode.

Здесь он просто зависает и в конечном итоге истекает время


person Kevin    schedule 19.01.2010    source источник


Ответы (3)


Наконец, возникла проблема с загрузкой конфигурации log4j с конфигурацией JMS, когда приложение JMS устанавливало соединение. Если я загружаю конфигурацию без определенных уровней журнала ActiveMQ или приложений, проблема не возникает после подключения JMSAppender, тогда я могу загрузить дополнительную конфигурацию, чтобы позволить ей войти в JMS.

person Kevin    schedule 01.05.2010

Поскольку вы не предоставили подробностей (о конфигурации, журналах, трассировках и т. д.), могу ли я спросить, следили ли вы за Как использовать приложение JMS log4j с ActiveMQ. Если да, и если этот образец работал, какие изменения вы внесли.

В качестве примечания: вместо log4j я бы рассмотрел возможность использования logback, его преемника. Проверьте его дополнения.

person Pascal Thivent    schedule 20.01.2010
comment
Точно следовал этому примеру, и с ActiveMQ 5.3, где он поставляется с кодом, он не работает из коробки. Я посмотрю на logback, но одним из моих основных критериев для этой работы является то, что я должен работать прозрачно с существующим кодом, в котором уже есть журналирование log4J. Я попытался подключить slf4J, но это не решило проблему с ActiveMQ. Я добавил некоторый пример кода, конфигурации и вывода журнала к исходному вопросу, теперь, когда у меня снова есть код - person Kevin; 20.01.2010

Вы можете попробовать использовать AsynchAppender, который позволяет потокам log4j НЕ блокироваться (т.е. зависать) в операторе журнала, если есть ошибка. Я смог сделать это для JMSAppender и консольного приложения как в AsynAppender. Однако вам необходимо изменить файл log4j.properties на файл конфигурации log4j.xml. формат.

person joe blow    schedule 01.03.2010