1 /* 2 * Copyright (c) 2017 Mockito contributors 3 * This program is made available under the terms of the MIT License. 4 */ 5 package org.mockitoutil; 6 7 import java.util.concurrent.atomic.AtomicBoolean; 8 import org.assertj.core.api.Assertions; 9 import org.junit.Test; 10 import org.mockito.Mockito; 11 12 import static org.assertj.core.api.Assertions.assertThat; 13 import static org.junit.Assert.fail; 14 import static org.mockitoutil.ClassLoaders.currentClassLoader; 15 import static org.mockitoutil.ClassLoaders.excludingClassLoader; 16 import static org.mockitoutil.ClassLoaders.isolatedClassLoader; 17 import static org.mockitoutil.ClassLoaders.jdkClassLoader; 18 19 public class ClassLoadersTest { 20 21 public static final String CLASS_NAME_DEPENDING_ON_INTERFACE = "org.mockitoutil.ClassLoadersTest$ClassUsingInterface1"; 22 public static final String INTERFACE_NAME = "org.mockitoutil.ClassLoadersTest$Interface1"; 23 24 @Test(expected = ClassNotFoundException.class) 25 public void isolated_class_loader_cannot_load_classes_when_no_given_prefix() throws Exception { 26 // given 27 ClassLoader cl = isolatedClassLoader().build(); 28 29 // when 30 cl.loadClass("org.mockito.Mockito"); 31 32 // then raises CNFE 33 } 34 35 @Test 36 public void isolated_class_loader_cannot_load_classes_if_no_code_source_path() throws Exception { 37 // given 38 ClassLoader cl = isolatedClassLoader() 39 .withPrivateCopyOf(CLASS_NAME_DEPENDING_ON_INTERFACE) 40 .build(); 41 42 // when 43 try { 44 cl.loadClass(CLASS_NAME_DEPENDING_ON_INTERFACE); 45 fail(); 46 } catch (ClassNotFoundException e) { 47 // then 48 assertThat(e).hasMessageContaining(CLASS_NAME_DEPENDING_ON_INTERFACE); 49 } 50 } 51 52 @Test 53 public void isolated_class_loader_cannot_load_classes_if_dependent_classes_do_not_match_the_prefixes() throws Exception { 54 // given 55 ClassLoader cl = isolatedClassLoader() 56 .withCurrentCodeSourceUrls() 57 .withPrivateCopyOf(CLASS_NAME_DEPENDING_ON_INTERFACE) 58 .build(); 59 60 // when 61 try { 62 cl.loadClass(CLASS_NAME_DEPENDING_ON_INTERFACE); 63 fail(); 64 } catch (NoClassDefFoundError e) { 65 // then 66 assertThat(e).hasMessageContaining("org/mockitoutil/ClassLoadersTest$Interface1"); 67 } 68 } 69 70 @Test 71 public void isolated_class_loader_can_load_classes_when_dependent_classes_are_matching_the_prefixes() throws Exception { 72 // given 73 ClassLoader cl = isolatedClassLoader() 74 .withCurrentCodeSourceUrls() 75 .withPrivateCopyOf(CLASS_NAME_DEPENDING_ON_INTERFACE) 76 .withPrivateCopyOf(INTERFACE_NAME) 77 .build(); 78 79 // when 80 Class<?> aClass = cl.loadClass(CLASS_NAME_DEPENDING_ON_INTERFACE); 81 82 // then 83 assertThat(aClass).isNotNull(); 84 assertThat(aClass.getClassLoader()).isEqualTo(cl); 85 assertThat(aClass.getInterfaces()[0].getClassLoader()).isEqualTo(cl); 86 } 87 88 @Test 89 public void isolated_class_loader_can_load_classes_isolated_classes_in_isolation() throws Exception { 90 // given 91 ClassLoader cl = isolatedClassLoader() 92 .withCurrentCodeSourceUrls() 93 .withPrivateCopyOf(ClassLoadersTest.class.getPackage().getName()) 94 .build(); 95 96 // when 97 Class<?> aClass = cl.loadClass(AClass.class.getName()); 98 99 // then 100 assertThat(aClass).isNotNull(); 101 assertThat(aClass).isNotSameAs(AClass.class); 102 assertThat(aClass.getClassLoader()).isEqualTo(cl); 103 } 104 105 @Test 106 public void isolated_class_loader_cannot_load_classes_if_prefix_excluded() throws Exception { 107 // given 108 ClassLoader cl = isolatedClassLoader() 109 .withCurrentCodeSourceUrls() 110 .withPrivateCopyOf(ClassLoadersTest.class.getPackage().getName()) 111 .without(AClass.class.getName()) 112 .build(); 113 114 // when 115 try { 116 cl.loadClass(AClass.class.getName()); 117 fail(); 118 } catch (ClassNotFoundException e) { 119 // then 120 assertThat(e).hasMessageContaining("org.mockitoutil") 121 .hasMessageContaining(AClass.class.getName()); 122 } 123 } 124 125 @Test 126 public void isolated_class_loader_has_no_parent() throws Exception { 127 ClassLoader cl = isolatedClassLoader() 128 .withCurrentCodeSourceUrls() 129 .withPrivateCopyOf(CLASS_NAME_DEPENDING_ON_INTERFACE) 130 .withPrivateCopyOf(INTERFACE_NAME) 131 .build(); 132 133 assertThat(cl.getParent()).isNull(); 134 } 135 136 @Test(expected = ClassNotFoundException.class) 137 public void excluding_class_loader_cannot_load_classes_when_no_correct_source_url_set() throws Exception { 138 // given 139 ClassLoader cl = excludingClassLoader() 140 .withCodeSourceUrlOf(this.getClass()) 141 .build(); 142 143 // when 144 cl.loadClass("org.mockito.Mockito"); 145 146 // then class CNFE 147 } 148 149 @Test 150 public void excluding_class_loader_can_load_classes_when_correct_source_url_set() throws Exception { 151 // given 152 ClassLoader cl = excludingClassLoader() 153 .withCodeSourceUrlOf(Mockito.class) 154 .build(); 155 156 // when 157 cl.loadClass("org.mockito.Mockito"); 158 159 // then class successfully loaded 160 } 161 162 @Test 163 public void excluding_class_loader_cannot_load_class_when_excluded_prefix_match_class_to_load() throws Exception { 164 // given 165 ClassLoader cl = excludingClassLoader() 166 .withCodeSourceUrlOf(Mockito.class) 167 .without("org.mockito.BDDMockito") 168 .build(); 169 170 cl.loadClass("org.mockito.Mockito"); 171 172 // when 173 try { 174 cl.loadClass("org.mockito.BDDMockito"); 175 fail("should have raise a ClassNotFoundException"); 176 } catch (ClassNotFoundException e) { 177 assertThat(e.getMessage()).contains("org.mockito.BDDMockito"); 178 } 179 180 // then class successfully loaded 181 } 182 183 @Test 184 public void can_not_load_a_class_not_previously_registered_in_builder() throws Exception { 185 // given 186 ClassLoader cl = ClassLoaders 187 .inMemoryClassLoader() 188 .withClassDefinition("yop.Dude", SimpleClassGenerator.makeMarkerInterface("yop.Dude")) 189 .build(); 190 191 // when 192 try { 193 cl.loadClass("not.Defined"); 194 fail(); 195 } catch (ClassNotFoundException e) { 196 // then 197 assertThat(e.getMessage()).contains("not.Defined"); 198 } 199 } 200 201 @Test 202 public void can_load_a_class_in_memory_from_bytes() throws Exception { 203 // given 204 ClassLoader cl = ClassLoaders 205 .inMemoryClassLoader() 206 .withClassDefinition("yop.Dude", SimpleClassGenerator.makeMarkerInterface("yop.Dude")) 207 .build(); 208 209 // when 210 Class<?> aClass = cl.loadClass("yop.Dude"); 211 212 // then 213 assertThat(aClass).isNotNull(); 214 assertThat(aClass.getClassLoader()).isEqualTo(cl); 215 assertThat(aClass.getName()).isEqualTo("yop.Dude"); 216 } 217 218 @Test 219 public void cannot_load_a_class_file_not_in_parent() throws Exception { 220 // given 221 ClassLoader cl = ClassLoaders 222 .inMemoryClassLoader() 223 .withParent(jdkClassLoader()) 224 .build(); 225 226 cl.loadClass("java.lang.String"); 227 228 try { 229 // when 230 cl.loadClass("org.mockito.Mockito"); 231 fail("should have not found Mockito class"); 232 } catch (ClassNotFoundException e) { 233 // then 234 assertThat(e.getMessage()).contains("org.mockito.Mockito"); 235 } 236 } 237 238 @Test 239 public void can_list_all_classes_reachable_in_a_classloader() throws Exception { 240 ClassLoader classLoader = ClassLoaders.inMemoryClassLoader() 241 .withParent(jdkClassLoader()) 242 .withClassDefinition("a.A", SimpleClassGenerator.makeMarkerInterface("a.A")) 243 .withClassDefinition("a.b.B", SimpleClassGenerator.makeMarkerInterface("a.b.B")) 244 .withClassDefinition("c.C", SimpleClassGenerator.makeMarkerInterface("c.C")) 245 // .withCodeSourceUrlOf(ClassLoaders.class) 246 .build(); 247 248 assertThat(ClassLoaders.in(classLoader).listOwnedClasses()).containsOnly("a.A", "a.b.B", "c.C"); 249 assertThat(ClassLoaders.in(classLoader).omit("b", "c").listOwnedClasses()).containsOnly("a.A"); 250 } 251 252 @Test 253 public void return_bootstrap_classloader() throws Exception { 254 assertThat(jdkClassLoader()).isNotEqualTo(Mockito.class.getClassLoader()); 255 assertThat(jdkClassLoader()).isNotEqualTo(ClassLoaders.class.getClassLoader()); 256 assertThat(jdkClassLoader()).isEqualTo(Number.class.getClassLoader()); 257 assertThat(jdkClassLoader()).isEqualTo(null); 258 } 259 260 @Test 261 public void return_current_classloader() throws Exception { 262 assertThat(currentClassLoader()).isEqualTo(this.getClass().getClassLoader()); 263 } 264 265 @Test 266 public void can_run_in_given_classloader() throws Exception { 267 // given 268 final ClassLoader cl = isolatedClassLoader() 269 .withCurrentCodeSourceUrls() 270 .withCodeSourceUrlOf(Assertions.class) 271 .withPrivateCopyOf("org.assertj.core") 272 .withPrivateCopyOf(ClassLoadersTest.class.getPackage().getName()) 273 .without(AClass.class.getName()) 274 .build(); 275 276 final AtomicBoolean executed = new AtomicBoolean(false); 277 278 // when 279 ClassLoaders.using(cl).execute(new Runnable() { 280 @Override 281 public void run() { 282 assertThat(this.getClass().getClassLoader()).describedAs("runnable is reloaded in given classloader").isEqualTo(cl); 283 assertThat(Thread.currentThread().getContextClassLoader()).describedAs("Thread context classloader is using given classloader").isEqualTo(cl); 284 285 try { 286 assertThat(Thread.currentThread() 287 .getContextClassLoader() 288 .loadClass("java.lang.String")) 289 .describedAs("can load JDK type") 290 .isNotNull(); 291 assertThat(Thread.currentThread() 292 .getContextClassLoader() 293 .loadClass("org.mockitoutil.ClassLoadersTest$ClassUsingInterface1")) 294 .describedAs("can load classloader types") 295 .isNotNull(); 296 } catch (ClassNotFoundException cnfe) { 297 Assertions.fail("should not have raised a CNFE", cnfe); 298 } 299 executed.set(true); 300 } 301 }); 302 303 // then 304 assertThat(executed.get()).isEqualTo(true); 305 } 306 307 308 @Test 309 public void cannot_load_runnable_in_given_classloader_if_some_type_cant_be_loaded() throws Exception { 310 // given 311 final ClassLoader cl = isolatedClassLoader() 312 .withCurrentCodeSourceUrls() 313 .withPrivateCopyOf(ClassLoadersTest.class.getPackage().getName()) 314 .without(AClass.class.getName()) 315 .build(); 316 317 // when 318 try { 319 ClassLoaders.using(cl).execute(new Runnable() { 320 @Override 321 public void run() { 322 AClass cant_be_found = new AClass(); 323 } 324 }); 325 Assertions.fail("should have raised a ClassNotFoundException"); 326 } catch (IllegalStateException ise) { 327 // then 328 assertThat(ise).hasCauseInstanceOf(NoClassDefFoundError.class) 329 .hasMessageContaining("AClass"); 330 } 331 } 332 333 @SuppressWarnings("unused") 334 static class AClass { 335 } 336 337 @SuppressWarnings("unused") 338 static class ClassUsingInterface1 implements Interface1 { 339 } 340 341 @SuppressWarnings("unused") 342 interface Interface1 { 343 } 344 } 345