blob: 502f4ca9a0adf15a33070bec369353f4ef61f91b [file] [log] [blame]
/*
* Copyright (c) 2012, Paul Merlin. All Rights Reserved.
*
* 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.apache.zest.runtime.mixin;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.json.JSONObject;
import org.junit.Before;
import org.junit.Test;
import org.apache.zest.api.common.Visibility;
import org.apache.zest.api.concern.Concerns;
import org.apache.zest.api.concern.GenericConcern;
import org.apache.zest.api.injection.scope.This;
import org.apache.zest.api.service.ServiceReference;
import org.apache.zest.bootstrap.AssemblyException;
import org.apache.zest.bootstrap.ModuleAssembly;
import org.apache.zest.functional.Specification;
import org.apache.zest.test.AbstractQi4jTest;
import static org.hamcrest.core.IsEqual.*;
import static org.junit.Assert.*;
import static org.apache.zest.functional.Iterables.*;
/**
* Assert that JDK classes are usable as Mixins.
*/
public class JDKMixinTest
extends AbstractQi4jTest
{
@Concerns( JDKMixinConcern.class )
public interface JSONSerializableMap
extends Map<String, String>
{
JSONObject toJSON();
}
@SuppressWarnings( "serial" )
public static class ExtendsJDKMixin
extends HashMap<String, String>
implements JSONSerializableMap
{
@Override
public JSONObject toJSON()
{
System.out.println( ">>>> Call ExtendsJDKMixin.toJSON()" );
// Copy the Map before handing it to JSONObject so that the JSONObject do not use the Composite
return new JSONObject( new HashMap<String, String>( this ) );
}
}
public static abstract class ComposeWithJDKMixin
implements JSONSerializableMap
{
@This
private Map<String, String> map;
@Override
public JSONObject toJSON()
{
System.out.println( ">>>> Call ComposeWithJDKMixin.toJSON()" );
// Copy the Map before handing it to JSONObject so that the JSONObject do not use the Composite
return new JSONObject( new HashMap<String, String>( map ) );
}
}
public static class JDKMixinConcern
extends GenericConcern
{
@Override
public Object invoke( Object proxy, Method method, Object[] args )
throws Throwable
{
System.out.println( ">>>> Call to JDKMixinConcern." + method.getName() );
CONCERN_RECORDS.add( method.getName() );
return next.invoke( proxy, method, args );
}
}
private static final String EXTENDS_IDENTITY = ExtendsJDKMixin.class.getName();
private static final String COMPOSE_IDENTITY = ComposeWithJDKMixin.class.getName();
private static final Specification<ServiceReference<?>> EXTENDS_IDENTITY_SPEC = new ServiceIdentitySpec( EXTENDS_IDENTITY );
private static final Specification<ServiceReference<?>> COMPOSE_IDENTITY_SPEC = new ServiceIdentitySpec( COMPOSE_IDENTITY );
private static final List<String> CONCERN_RECORDS = new ArrayList<String>();
@Before
public void beforeEachTest()
{
CONCERN_RECORDS.clear();
}
@Override
public void assemble( ModuleAssembly module )
throws AssemblyException
{
module.services( JSONSerializableMap.class ).
identifiedBy( EXTENDS_IDENTITY ).
withMixins( ExtendsJDKMixin.class ).
instantiateOnStartup();
module.layer().module( "compose" ).services( JSONSerializableMap.class ).
visibleIn( Visibility.layer ).
identifiedBy( COMPOSE_IDENTITY ).
withMixins( HashMap.class, ComposeWithJDKMixin.class ).
instantiateOnStartup();
}
@Test
public void testMixinExtendsJDK()
{
List<ServiceReference<JSONSerializableMap>> services = toList(
filter( EXTENDS_IDENTITY_SPEC,
module.findServices( JSONSerializableMap.class ) ) );
assertThat( services.size(), equalTo( 1 ) );
assertThat( services.get( 0 ).identity(), equalTo( EXTENDS_IDENTITY ) );
JSONSerializableMap extending = services.get( 0 ).get();
extending.put( "foo", "bar" ); // Concern trigger #1 (put)
JSONObject json = extending.toJSON(); // Concern trigger #2, #3 and #4 (toJSON, size, entrySet)
assertThat( json.length(), equalTo( 1 ) );
assertThat( json.optString( "foo" ), equalTo( "bar" ) );
assertThat( CONCERN_RECORDS.size(), equalTo( 4 ) );
}
@Test
public void testComposeJDKMixin()
{
List<ServiceReference<JSONSerializableMap>> services = toList(
filter( COMPOSE_IDENTITY_SPEC,
module.findServices( JSONSerializableMap.class ) ) );
assertThat( services.size(), equalTo( 1 ) );
assertThat( services.get( 0 ).identity(), equalTo( COMPOSE_IDENTITY ) );
JSONSerializableMap composing = services.get( 0 ).get();
composing.put( "foo", "bar" ); // Concern trigger #1 (put)
JSONObject json = composing.toJSON(); // Concern trigger #2, #3 and #4 (toJSON, size, entrySet)
assertThat( json.length(), equalTo( 1 ) );
assertThat( json.optString( "foo" ), equalTo( "bar" ) );
assertThat( CONCERN_RECORDS.size(), equalTo( 4 ) );
}
private static class ServiceIdentitySpec
implements Specification<ServiceReference<?>>
{
private final String identity;
public ServiceIdentitySpec( String identity )
{
this.identity = identity;
}
@Override
public boolean satisfiedBy( ServiceReference<?> item )
{
return item.identity().equals( identity );
}
}
}