1 /**
2 * Copyright 2005-2015 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.core.api.util;
17
18 import java.util.ArrayList;
19 import java.util.List;
20 import java.util.concurrent.Callable;
21
22 /**
23 * Utility class that binds/unbinds the context ClassLoader of the current Thread
24 * using a ThreadLocal.
25 * NOTE: {@link #doInContextClassLoader(ClassLoader, java.util.concurrent.Callable)} is the only safe way to use this class,
26 * do not use deprecated {@link #bind(ClassLoader)} or {@link #unbind()} methods.
27 *
28 * @author Kuali Rice Team (rice.collab@kuali.org)
29 */
30 public final class ContextClassLoaderBinder {
31
32 /**
33 * Stack of previous context classloaders that should be
34 * restored on unbind
35 * @deprecated since 2.1, use #doInContextClassLoader. storing references to classloaders is an incredibly bad idea!
36 */
37 private static final ThreadLocal<List<ClassLoader>> STACK = new ThreadLocal<List<ClassLoader>>() {
38 protected List<ClassLoader> initialValue() {
39 return new ArrayList<ClassLoader>(5);
40 }
41 };
42
43 /**
44 * @deprecated use #doInContextClassLoader
45 */
46 private static List<ClassLoader> getStack() {
47 return STACK.get();
48 }
49
50 /**
51 * @deprecated use #doInContextClassLoader
52 */
53 public static void bind(ClassLoader cl) {
54 List<ClassLoader> stack = getStack();
55 Thread current = Thread.currentThread();
56 //log.debug("[bind] Switching CCL from " + current.getContextClassLoader() + " to " + cl);
57 // push the current context classloader on the stack
58 stack.add(current.getContextClassLoader());
59 current.setContextClassLoader(cl);
60 }
61
62 /**
63 * @deprecated use #doInContextClassLoader
64 */
65 public static void unbind() {
66 List<ClassLoader> stack = getStack();
67 if (stack.size() == 0) {
68 throw new IllegalStateException("No context classloader to unbind!");
69 }
70 // pop the last context classloader off the stack
71 ClassLoader lastClassLoader = stack.get(stack.size() - 1);
72 //log.debug("[unbind] Switching CCL from " + Thread.currentThread().getContextClassLoader() + " to " + lastClassLoader);
73 stack.remove(stack.size() - 1);
74 Thread.currentThread().setContextClassLoader(lastClassLoader);
75 }
76
77 /**
78 * Execute a runnable under a given context classloader
79 * @param cl the classloader to set as the thread context classloader
80 * @param callable the callable
81 */
82 public static <T> T doInContextClassLoader(ClassLoader cl, Callable<T> callable) throws Exception {
83 Thread current = Thread.currentThread();
84 ClassLoader prev = current.getContextClassLoader();
85 try {
86 current.setContextClassLoader(cl);
87 return callable.call();
88 } finally {
89 current.setContextClassLoader(prev);
90 }
91 }
92 }