Skip to content

Commit

Permalink
Protect against non-deterministic method order in JDK7
Browse files Browse the repository at this point in the history
 - Allow reset of GlobalAdvisorAdapterRegistry

   Provide a reset() method allowing the GlobalAdvisorAdapterRegistry
   instance to be replaced with a fresh instance. This method has
   primarily been added to allow unit tests to leave the registry
   in a known state.

 - Protect against the fact that calls to configuration class methods
   my occur in a random order.

Issue: SPR-9779
  • Loading branch information
philwebb authored and cbeams committed Sep 11, 2012
1 parent 8e7622b commit a9a90ca
Show file tree
Hide file tree
Showing 3 changed files with 50 additions and 33 deletions.
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
/*
* Copyright 2002-2005 the original author or authors.
*
* Copyright 2002-2012 the original author or authors.
*
* 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.
Expand All @@ -21,20 +21,30 @@
*
* @author Rod Johnson
* @author Juergen Hoeller
* @author Phillip Webb
* @see DefaultAdvisorAdapterRegistry
*/
public abstract class GlobalAdvisorAdapterRegistry {

/**
* Keep track of a single instance so we can return it to classes that request it.
*/
private static final AdvisorAdapterRegistry instance = new DefaultAdvisorAdapterRegistry();
private static AdvisorAdapterRegistry instance = new DefaultAdvisorAdapterRegistry();

/**
* Return the singleton DefaultAdvisorAdapterRegistry instance.
* Return the singleton {@link DefaultAdvisorAdapterRegistry} instance.
*/
public static AdvisorAdapterRegistry getInstance() {
return instance;
}

/**
* Reset the singleton {@link DefaultAdvisorAdapterRegistry}, removing any
* {@link AdvisorAdapterRegistry#registerAdvisorAdapter(AdvisorAdapter) registered}
* adapters.
*/
static void reset() {
instance = new DefaultAdvisorAdapterRegistry();
}

}
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
/*
* Copyright 2002-2005 the original author or authors.
*
* Copyright 2002-2012 the original author or authors.
*
* 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.
Expand All @@ -23,6 +23,8 @@
import org.aopalliance.aop.Advice;
import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.springframework.aop.Advisor;
import org.springframework.aop.BeforeAdvice;
Expand All @@ -38,6 +40,12 @@
*/
public final class AdvisorAdapterRegistrationTests {

@Before
@After
public void resetGlobalAdvisorAdapterRegistry() {
GlobalAdvisorAdapterRegistry.reset();
}

@Test
public void testAdvisorAdapterRegistrationManagerNotPresentInContext() {
ClassPathXmlApplicationContext ctx =
Expand Down Expand Up @@ -101,7 +109,7 @@ public MethodInterceptor getInterceptor(Advisor advisor) {


class SimpleBeforeAdviceImpl implements SimpleBeforeAdvice {

private int invocationCounter;

public void before() throws Throwable {
Expand All @@ -116,9 +124,9 @@ public int getInvocationCounter() {


final class SimpleBeforeAdviceInterceptor implements MethodInterceptor {

private SimpleBeforeAdvice advice;

public SimpleBeforeAdviceInterceptor(SimpleBeforeAdvice advice) {
this.advice = advice;
}
Expand All @@ -127,5 +135,4 @@ public Object invoke(MethodInvocation mi) throws Throwable {
advice.before();
return mi.proceed();
}

}
}
Original file line number Diff line number Diff line change
Expand Up @@ -16,11 +16,13 @@

package org.springframework.test.context.junit4.spr9051;

import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertTrue;
import static org.hamcrest.Matchers.*;
import static org.hamcrest.Matchers.is;
import static org.junit.Assert.*;

import java.util.Arrays;
import java.util.HashSet;
import java.util.Set;
import java.util.concurrent.atomic.AtomicInteger;

import org.junit.Test;
Expand All @@ -30,22 +32,24 @@
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;


/**
* This set of tests refutes the claims made in
* <a href="https://jira.springsource.org/browse/SPR-9051" target="_blank">SPR-9051</a>.
*
*
* <p><b>The Claims</b>:
*
*
* <blockquote>
* When a {@code @ContextConfiguration} test class references a config class
* missing an {@code @Configuration} annotation, {@code @Bean} dependencies are
* wired successfully but the bean lifecycle is not applied (no init methods are
* invoked, for example). Adding the missing {@code @Configuration} annotation
* solves the problem, however the problem and solution isn't obvious since
* solves the problem, however the problem and solution isn't obvious since
* wiring/injection appeared to work.
* </blockquote>
*
*
* @author Sam Brannen
* @author Phillip Webb
* @since 3.2
*/
@RunWith(SpringJUnit4ClassRunner.class)
Expand Down Expand Up @@ -85,18 +89,14 @@ public LifecycleBean lifecycleBean() {
@Autowired
private LifecycleBean lifecycleBean;


@Test
public void simpleStringBean() {
public void testSPR_9051() throws Exception {
assertNotNull(enigma);
assertEquals("enigma #1", enigma);
}

@Test
public void beanWithLifecycleCallback() {
assertNotNull(lifecycleBean);
assertEquals("enigma #2", lifecycleBean.getName());
assertTrue(lifecycleBean.isInitialized());
Set<String> names = new HashSet<String>();
names.add(enigma.toString());
names.add(lifecycleBean.getName());
assertEquals(names, new HashSet<String>(Arrays.asList("enigma #1", "enigma #2")));
}

}

0 comments on commit a9a90ca

Please sign in to comment.