javamultithreadinglog4jquartz-schedulerscheduler

Splitting log4j Output with Quartz Worker Threads


I'm working on an application that consists of an overall Quartz-based scheduler and "CycledJob" run using CronTriggers. The purpose of the application is to process inputs from different email inboxes based on the source country.

Based on the country that it comes in from (i.e. US, UK, FR, etc.) the application triggers one job thread to run each country's processing cycle, so there would be a UK Worker thread, one for US, France, etc. When formatting the output to log4j, I'm using the thread parameter, so it emits [ApplicationName_Worker-1], [ApplicationName_Worker-2] etc. Try as I might, I can't find a way to name the threads since they're pulled out of Quartz's Thread Pools. Although I could possibly go so far as to extend Quartz, I'd like to work out a different solution instead of messing with the standard library.

Here's the problem: When using log4j, I'd like to have all log items from the US thread output to a US only file, likewise for each of the country threads. I don't care if they stay in one unified ConsoleAppender, the FileAppender split is what I'm after here. I already know how to specify multiple file appenders and such, my issue is I can't differentiate based on country.

There are 20+ classes within the application that can be on the execution chain, very few of which I want to burden with the knowledge of passing an extra "context" parameter through EVERY method... I've considered a Strategy pattern extending a log4j wrapper class, but unless I can let every class in the chain know which thread it's on to parameterize the logger call, that seems impossible. Without being able to name the thread also creates a challenge (or else this would be easy!).

So here's the question: What would be a suggested approach to allow many subordinate classes in an application that are each used for every different thread to process the input know that they are within the context of a particular country thread when they are logging?


Solution

  • At the top of each country's processing thread, put the country code into Log4j's mapped diagnostic context (MDC). This uses a ThreadLocal variable so that you don't have to pass the country up and down the call stack explicitly. Then create a custom filter that looks at the MDC, and filters out any events that don't contain the current appender's country code.

    In your Job:

    ...
    public static final String MDC_COUNTRY = "com.y.foo.Country";
    public void execute(JobExecutionContext context)
      /* Just guessing that you have the country in your JobContext. */
      MDC.put(MDC_COUNTRY, context.get(MDC_COUNTRY));
      try {
        /* Perform your job here. */
        ...
      } finally {
        MDC.remove(MDC_COUNTRY);
      }
    }
    ...
    

    Write a custom Filter:

    package com.y.log4j;
    
    import org.apache.log4j.spi.LoggingEvent;
    
    /**
     * This is a general purpose filter. If its "value" property is null, 
     * it requires only that the specified key be set in the MDC. If its 
     * value is not null, it further requires that the value in the MDC 
     * is equal.
     */
    public final class ContextFilter extends org.apache.log4j.spi.Filter {
    
      public int decide(LoggingEvent event) {
        Object ctx = event.getMDC(key);
        if (value == null)
          return (ctx != null) ? NEUTRAL : DENY;
        else
          return value.equals(ctx) ? NEUTRAL : DENY;
      }
    
      private String key;
      private String value;
    
      public void setContextKey(String key) { this.key = key; }
      public String getContextKey() { return key; }
      public void setValue(String value) { this.value = value; }
      public String getValue() { return value; }
    
    }
    

    In your log4j.xml:

    <appender name="fr" class="org.apache.log4j.FileAppender">
      <param name="file" value="france.log"/>
      ...
      <filter class="com.y.log4j.ContextFilter">
        <param name="key" value="com.y.foo.Country" />
        <param name="value" value="fr" />
      </filter>
    </appender>