View Javadoc

1   /**
2    * Copyright 2005-2012 The Kuali Foundation
3    *
4    * Licensed under the Educational Community License, Version 2.0 (the "License");
5    * you may not use this file except in compliance with the License.
6    * You may obtain a copy of the License at
7    *
8    * http://www.opensource.org/licenses/ecl2.php
9    *
10   * Unless required by applicable law or agreed to in writing, software
11   * distributed under the License is distributed on an "AS IS" BASIS,
12   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13   * See the License for the specific language governing permissions and
14   * limitations under the License.
15   */
16  package org.kuali.rice.ksb.util;
17  
18  import org.apache.log4j.Logger;
19  import org.apache.xml.security.algorithms.SignatureAlgorithm;
20  import org.kuali.rice.core.framework.util.ApplicationThreadLocal;
21  import org.springframework.beans.BeansException;
22  import org.springframework.context.ApplicationContext;
23  import org.springframework.context.ApplicationContextAware;
24  import org.springframework.context.ApplicationListener;
25  import org.springframework.context.ConfigurableApplicationContext;
26  import org.springframework.context.event.ContextClosedEvent;
27  
28  import java.lang.reflect.Field;
29  import java.util.ArrayList;
30  import java.util.Collection;
31  
32  /**
33   * Removes {@link SignatureAlgorithm} ThreadLocals
34   */
35  public class SignatureAlgorithmCacheCleaner implements ApplicationContextAware {
36      private static final Logger LOG = Logger.getLogger(SignatureAlgorithmCacheCleaner.class);
37  
38      /**
39       * Registers a listener that removes SignatureAlgorithm key caches on context shutdown
40       * @param applicationContext spring context
41       */
42      @Override
43      public void setApplicationContext(ApplicationContext applicationContext) {
44          // register a context close handler
45          if (applicationContext instanceof ConfigurableApplicationContext) {
46              ConfigurableApplicationContext context = (ConfigurableApplicationContext) applicationContext;
47              context.addApplicationListener(new ApplicationListener<ContextClosedEvent>() {
48                  @Override
49                  public synchronized void onApplicationEvent(ContextClosedEvent e) {
50                      LOG.info("Context '" + e.getApplicationContext().getDisplayName() + "' closed, removing SignatureAlgorithm caches");
51  
52                      // obtain the thread local caches via reflection
53                      String[] threadLocalFields = new String[] { "instancesSigning", "instancesVerify", "keysSigning", "keysVerify" };
54                      Collection<ThreadLocal> threadLocals = new ArrayList<ThreadLocal>(threadLocalFields.length);
55                      for (String fname: threadLocalFields) {
56                          try {
57                              Field f = SignatureAlgorithm.class.getDeclaredField(fname);
58                              if (f != null) {
59                                  f.setAccessible(true);
60                                  ThreadLocal tl = (ThreadLocal) f.get(null);
61                                  if (tl != null) {
62                                      threadLocals.add(tl);
63                                  }
64                              }
65                          } catch (NoSuchFieldException nsme) {
66                              nsme.printStackTrace();
67                          } catch (IllegalAccessException iae) {
68                              iae.printStackTrace();
69                          }
70                      }
71  
72                      // remove the thread locals from every thread
73                      if (!threadLocals.isEmpty()) {
74                          for (Thread t: Thread.getAllStackTraces().keySet()) {
75                              for (ThreadLocal tl: threadLocals) {
76                                  ApplicationThreadLocal.removeThreadLocal(t, tl);
77                              }
78                          }
79                      }
80                  }
81              });
82          }
83      }
84  }