CachingDestinationResolverProxy for slow target DestinationResolvers
Issue: SPR-11939
This commit is contained in:
parent
ea4a5d4722
commit
c84b30d4a4
|
@ -0,0 +1,94 @@
|
|||
/*
|
||||
* Copyright 2002-2014 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.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package org.springframework.messaging.core;
|
||||
|
||||
import java.util.concurrent.ConcurrentHashMap;
|
||||
|
||||
import org.springframework.beans.factory.InitializingBean;
|
||||
import org.springframework.util.Assert;
|
||||
|
||||
/**
|
||||
* {@link DestinationResolver} implementation that proxies a target DestinationResolver,
|
||||
* caching its {@link #resolveDestination} results. Such caching is particularly useful
|
||||
* if the destination resolving process is expensive (e.g. the destination has to be
|
||||
* resolved through an external system) and the resolution results are stable anyway.
|
||||
*
|
||||
* @author Agim Emruli
|
||||
* @author Juergen Hoeller
|
||||
* @since 4.1
|
||||
* @see DestinationResolver#resolveDestination
|
||||
*/
|
||||
public class CachingDestinationResolverProxy<D> implements DestinationResolver<D>, InitializingBean {
|
||||
|
||||
private final ConcurrentHashMap<String, D> resolvedDestinationCache = new ConcurrentHashMap<String, D>();
|
||||
|
||||
private DestinationResolver<D> targetDestinationResolver;
|
||||
|
||||
|
||||
/**
|
||||
* Create a new CachingDestinationResolverProxy, setting the target DestinationResolver
|
||||
* through the {@link #setTargetDestinationResolver} bean property.
|
||||
*/
|
||||
public CachingDestinationResolverProxy() {
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a new CachingDestinationResolverProxy using the given target
|
||||
* DestinationResolver to actually resolve destinations.
|
||||
* @param targetDestinationResolver the target DestinationResolver to delegate to
|
||||
*/
|
||||
public CachingDestinationResolverProxy(DestinationResolver<D> targetDestinationResolver) {
|
||||
Assert.notNull(targetDestinationResolver, "Target DestinationResolver must not be null");
|
||||
this.targetDestinationResolver = targetDestinationResolver;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Set the target DestinationResolver to delegate to.
|
||||
*/
|
||||
public void setTargetDestinationResolver(DestinationResolver<D> targetDestinationResolver) {
|
||||
this.targetDestinationResolver = targetDestinationResolver;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void afterPropertiesSet() {
|
||||
if (this.targetDestinationResolver == null) {
|
||||
throw new IllegalArgumentException("Property 'targetDestinationResolver' is required");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
*
|
||||
* Resolves and caches destinations if successfully resolved by the target
|
||||
* DestinationResolver implementation.
|
||||
* @param name the destination name to be resolved
|
||||
* @return the currently resolved destination or an already cached destination
|
||||
* @throws DestinationResolutionException if the target DestinationResolver
|
||||
* reports an error during destination resolution
|
||||
*/
|
||||
@Override
|
||||
public D resolveDestination(String name) throws DestinationResolutionException {
|
||||
D destination = this.resolvedDestinationCache.get(name);
|
||||
if (destination == null) {
|
||||
destination = this.targetDestinationResolver.resolveDestination(name);
|
||||
this.resolvedDestinationCache.putIfAbsent(name, destination);
|
||||
}
|
||||
return destination;
|
||||
}
|
||||
|
||||
}
|
|
@ -0,0 +1,60 @@
|
|||
/*
|
||||
* Copyright 2002-2014 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.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
package org.springframework.messaging.core;
|
||||
|
||||
import org.junit.Test;
|
||||
|
||||
import static org.junit.Assert.*;
|
||||
import static org.mockito.Mockito.*;
|
||||
|
||||
/**
|
||||
* Unit tests for {@link CachingDestinationResolverProxy}.
|
||||
*
|
||||
* @author Agim Emruli
|
||||
* @author Juergen Hoeller
|
||||
*/
|
||||
public class CachingDestinationResolverTests {
|
||||
|
||||
@Test
|
||||
public void cachedDestination() {
|
||||
@SuppressWarnings("unchecked")
|
||||
DestinationResolver<String> destinationResolver = (DestinationResolver<String>) mock(DestinationResolver.class);
|
||||
CachingDestinationResolverProxy<String> cachingDestinationResolver = new CachingDestinationResolverProxy<String>(destinationResolver);
|
||||
|
||||
when(destinationResolver.resolveDestination("abcd")).thenReturn("dcba");
|
||||
when(destinationResolver.resolveDestination("1234")).thenReturn("4321");
|
||||
|
||||
assertEquals("dcba", cachingDestinationResolver.resolveDestination("abcd"));
|
||||
assertEquals("4321", cachingDestinationResolver.resolveDestination("1234"));
|
||||
assertEquals("4321", cachingDestinationResolver.resolveDestination("1234"));
|
||||
assertEquals("dcba", cachingDestinationResolver.resolveDestination("abcd"));
|
||||
|
||||
verify(destinationResolver, times(1)).resolveDestination("abcd");
|
||||
verify(destinationResolver, times(1)).resolveDestination("1234");
|
||||
}
|
||||
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void noTargetSet() {
|
||||
CachingDestinationResolverProxy<String> cachingDestinationResolver = new CachingDestinationResolverProxy<String>();
|
||||
cachingDestinationResolver.afterPropertiesSet();
|
||||
}
|
||||
|
||||
@Test(expected = IllegalArgumentException.class)
|
||||
public void nullTargetThroughConstructor() {
|
||||
new CachingDestinationResolverProxy<String>(null);
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in New Issue