/*
* Copyright 2015 Google Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.google.template.soy.jbcsrc;
import static com.google.common.truth.Truth.assertThat;
import static com.google.template.soy.jbcsrc.SoyExpression.FALSE;
import static com.google.template.soy.jbcsrc.SoyExpression.TRUE;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import com.google.common.collect.ImmutableList;
import com.google.common.io.ByteStreams;
import com.google.common.testing.GcFinalization;
import com.google.template.soy.jbcsrc.ExpressionTester.BooleanInvoker;
import java.io.IOException;
import java.lang.ref.WeakReference;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.Type;
import org.objectweb.asm.tree.ClassNode;
/** Tests for {@link MemoryClassLoader} */
@RunWith(JUnit4.class)
public class MemoryClassLoaderTest {
// Our memory classloaders should be garbage collectable when all references to their types
// disapear
@Test
public void testCollectable() {
BooleanInvoker invoker = ExpressionTester.createInvoker(BooleanInvoker.class, FALSE);
assertFalse(invoker.invoke()); // sanity, the invoker works
MemoryClassLoader loader = (MemoryClassLoader) invoker.getClass().getClassLoader();
WeakReference<MemoryClassLoader> loaderRef = new WeakReference<MemoryClassLoader>(loader);
invoker = null; // unpin
loader = null;
GcFinalization.awaitClear(loaderRef);
}
@Test
public void testAsResource() throws IOException {
BooleanInvoker invoker = ExpressionTester.createInvoker(BooleanInvoker.class, FALSE);
byte[] classBytes =
ByteStreams.toByteArray(
invoker
.getClass()
.getClassLoader()
.getResourceAsStream(invoker.getClass().getName().replace('.', '/') + ".class"));
ClassNode node = new ClassNode();
new ClassReader(classBytes).accept(node, 0);
assertEquals(Type.getInternalName(invoker.getClass()), node.name);
}
@Test
public void testDelegation() throws Exception {
// We want to make sure that for generated types, our classloader doesn't delegate to its parent
// loader
ClassData parentClass = ExpressionTester.createClass(BooleanInvoker.class, FALSE);
ClassData childClass = ExpressionTester.createClass(BooleanInvoker.class, TRUE);
// sanity check, they have the same fully qualified class name
assertEquals(childClass.type(), parentClass.type());
TypeInfo type = childClass.type();
MemoryClassLoader parentLoader = new MemoryClassLoader(ImmutableList.of(parentClass));
MemoryClassLoader childLoader =
new MemoryClassLoader(parentLoader, ImmutableList.of(childClass));
BooleanInvoker fromParent =
parentLoader
.loadClass(type.className())
.asSubclass(BooleanInvoker.class)
.getConstructor()
.newInstance();
assertThat(fromParent.invoke()).isFalse();
BooleanInvoker fromChild =
childLoader
.loadClass(type.className())
.asSubclass(BooleanInvoker.class)
.getConstructor()
.newInstance();
assertThat(fromChild.invoke()).isTrue();
assertThat(fromChild.getClass()).isNotEqualTo(fromParent.getClass());
}
}