#1482 allowing generic self references in types when matching

This commit is contained in:
Sjaak Derksen 2018-07-14 16:27:10 +02:00 committed by GitHub
parent 616aaa986d
commit 58a2aa94bb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 413 additions and 11 deletions

View File

@ -276,6 +276,15 @@ public class MethodReference extends ModelElement implements Assignment {
return parameterBindings; return parameterBindings;
} }
public Type inferTypeWhenEnum( Type type ) {
if ( "java.lang.Enum".equals( type.getFullyQualifiedName() ) ) {
return type.getTypeParameters().get( 0 );
}
else {
return type;
}
}
@Override @Override
public int hashCode() { public int hashCode() {
final int prime = 31; final int prime = 31;

View File

@ -263,8 +263,9 @@ public class MethodMatcher {
// (type args are checked later). // (type args are checked later).
if ( p.getKind() == TypeKind.DECLARED ) { if ( p.getKind() == TypeKind.DECLARED ) {
DeclaredType t1 = (DeclaredType) p; DeclaredType t1 = (DeclaredType) p;
if ( assignabilityMatches( t, t1 ) if ( rawAssignabilityMatches( t, t1 ) ) {
&& t.getTypeArguments().size() == t1.getTypeArguments().size() ) { if ( t.getTypeArguments().size() == t1.getTypeArguments().size() ) {
// compare type var side by side
for ( int i = 0; i < t.getTypeArguments().size(); i++ ) { for ( int i = 0; i < t.getTypeArguments().size(); i++ ) {
if ( !visit( t.getTypeArguments().get( i ), t1.getTypeArguments().get( i ) ) ) { if ( !visit( t.getTypeArguments().get( i ), t1.getTypeArguments().get( i ) ) ) {
return Boolean.FALSE; return Boolean.FALSE;
@ -272,6 +273,13 @@ public class MethodMatcher {
} }
return Boolean.TRUE; return Boolean.TRUE;
} }
else {
// return true (e.g. matching Enumeration<E> with an enumeration E)
// but do not try to line up raw type arguments with types that do have arguments.
return assignability == Assignability.VISITED_ASSIGNABLE_TO ?
!t1.getTypeArguments().isEmpty() : !t.getTypeArguments().isEmpty();
}
}
else { else {
return Boolean.FALSE; return Boolean.FALSE;
} }
@ -284,12 +292,12 @@ public class MethodMatcher {
} }
} }
private boolean assignabilityMatches(DeclaredType visited, DeclaredType param) { private boolean rawAssignabilityMatches(DeclaredType t1, DeclaredType t2) {
if ( assignability == Assignability.VISITED_ASSIGNABLE_TO ) { if ( assignability == Assignability.VISITED_ASSIGNABLE_TO ) {
return typeUtils.isAssignable( toRawType( visited ), toRawType( param ) ); return typeUtils.isAssignable( toRawType( t1 ), toRawType( t2 ) );
} }
else { else {
return typeUtils.isAssignable( toRawType( param ), toRawType( visited ) ); return typeUtils.isAssignable( toRawType( t2 ), toRawType( t1 ) );
} }
} }
@ -302,7 +310,7 @@ public class MethodMatcher {
if ( genericTypesMap.containsKey( t ) ) { if ( genericTypesMap.containsKey( t ) ) {
// when already found, the same mapping should apply // when already found, the same mapping should apply
TypeMirror p1 = genericTypesMap.get( t ); TypeMirror p1 = genericTypesMap.get( t );
return typeUtils.isSameType( p, p1 ); return typeUtils.isSubtype( p, p1 );
} }
else { else {
// check if types are in bound // check if types are in bound

View File

@ -48,7 +48,7 @@
<#list parameterBindings as param> <#list parameterBindings as param>
<#if param.targetType> <#if param.targetType>
<#-- a class is passed on for casting, see @TargetType --> <#-- a class is passed on for casting, see @TargetType -->
<@includeModel object=ext.targetType raw=true/>.class<#t> <@includeModel object=inferTypeWhenEnum( ext.targetType ) raw=true/>.class<#t>
<#elseif param.mappingTarget> <#elseif param.mappingTarget>
${ext.targetBeanName}<#if ext.targetReadAccessorName??>.${ext.targetReadAccessorName}</#if><#t> ${ext.targetBeanName}<#if ext.targetReadAccessorName??>.${ext.targetReadAccessorName}</#if><#t>
<#elseif param.mappingContext> <#elseif param.mappingContext>

View File

@ -0,0 +1,35 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
import java.math.BigDecimal;
public class BigDecimalWrapper implements ValueWrapper<BigDecimal> {
private final BigDecimal value;
public BigDecimalWrapper(BigDecimal value) {
this.value = value;
}
@Override
public BigDecimal getValue() {
return value;
}
}

View File

@ -0,0 +1,75 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
import java.math.BigDecimal;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mapstruct.ap.testutil.IssueKey;
import org.mapstruct.ap.testutil.WithClasses;
import org.mapstruct.ap.testutil.runner.AnnotationProcessorTestRunner;
import static org.assertj.core.api.Assertions.assertThat;
@WithClasses({
Source.class,
Source2.class,
Target.class,
SourceEnum.class,
SourceTargetMapper.class,
TargetSourceMapper.class,
BigDecimalWrapper.class,
ValueWrapper.class
})
@IssueKey(value = "1482")
@RunWith(AnnotationProcessorTestRunner.class)
public class Issue1482Test {
@Test
public void testForward() {
Source source = new Source();
source.setTest( SourceEnum.VAL1 );
source.setWrapper( new BigDecimalWrapper( new BigDecimal( 5 ) ) );
Target target = SourceTargetMapper.INSTANCE.map( source );
assertThat( target ).isNotNull();
assertThat( target.getTest() ).isEqualTo( "value1" );
assertThat( target.getBigDecimal() ).isEqualTo( new BigDecimal( 5 ) );
}
@Test
public void testReverse() {
Target target = new Target();
target.setBigDecimal( new BigDecimal( 5 ) );
target.setTest( "VAL1" );
Source2 source2 = TargetSourceMapper.INSTANCE.map( target );
assertThat( source2 ).isNotNull();
assertThat( source2.getTest() ).isEqualTo( SourceEnum.VAL1 );
assertThat( source2.getWrapper().getValue() ).isEqualTo( new BigDecimal( 5 ) );
}
}

View File

@ -0,0 +1,41 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
public class Source {
private SourceEnum test;
private BigDecimalWrapper wrapper;
public SourceEnum getTest() {
return test;
}
public void setTest(SourceEnum test) {
this.test = test;
}
public BigDecimalWrapper getWrapper() {
return wrapper;
}
public void setWrapper(BigDecimalWrapper wrapper) {
this.wrapper = wrapper;
}
}

View File

@ -0,0 +1,43 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
import java.math.BigDecimal;
public class Source2 {
private Enum<SourceEnum> test;
private ValueWrapper<BigDecimal> wrapper;
public Enum<SourceEnum> getTest() {
return test;
}
public void setTest(Enum<SourceEnum> test) {
this.test = test;
}
public ValueWrapper<BigDecimal> getWrapper() {
return wrapper;
}
public void setWrapper(ValueWrapper<BigDecimal> wrapper) {
this.wrapper = wrapper;
}
}

View File

@ -0,0 +1,35 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
public enum SourceEnum {
VAL1( "value1" ),
VAL2( "value2" );
private final String val;
SourceEnum(String val) {
this.val = val;
}
public String toString() {
return val;
}
}

View File

@ -0,0 +1,40 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
import org.mapstruct.Mapper;
import org.mapstruct.Mapping;
import org.mapstruct.factory.Mappers;
@Mapper
public abstract class SourceTargetMapper {
public static final SourceTargetMapper INSTANCE = Mappers.getMapper( SourceTargetMapper.class );
@Mapping( target = "bigDecimal", source = "wrapper" )
abstract Target map(Source source);
protected String map(Enum<SourceEnum> e) {
return e.toString();
}
protected <T> T map(ValueWrapper<T> in) {
return in.getValue();
}
}

View File

@ -0,0 +1,43 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
import java.math.BigDecimal;
public class Target {
private String test;
private BigDecimal bigDecimal;
public String getTest() {
return test;
}
public void setTest(String test) {
this.test = test;
}
public BigDecimal getBigDecimal() {
return bigDecimal;
}
public void setBigDecimal(BigDecimal bigDecimal) {
this.bigDecimal = bigDecimal;
}
}

View File

@ -0,0 +1,50 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
import java.math.BigDecimal;
import org.mapstruct.Mapper;
import org.mapstruct.Mapping;
import org.mapstruct.TargetType;
import org.mapstruct.factory.Mappers;
@Mapper
public abstract class TargetSourceMapper {
public static final TargetSourceMapper INSTANCE = Mappers.getMapper( TargetSourceMapper.class );
@Mapping(target = "wrapper", source = "bigDecimal")
abstract Source2 map(Target target);
protected <T extends Enum<T>> Enum<T> map(String in, @TargetType Class<T>clz ) {
if ( clz.isAssignableFrom( SourceEnum.class )) {
return (Enum<T>) SourceEnum.valueOf( in );
}
return null;
}
protected <T> ValueWrapper<T> map(T in) {
if ( in instanceof BigDecimal ) {
return (ValueWrapper<T>) new BigDecimalWrapper( (BigDecimal) in );
}
return null;
}
}

View File

@ -0,0 +1,23 @@
/**
* Copyright 2012-2017 Gunnar Morling (http://www.gunnarmorling.de/)
* and/or other contributors as indicated by the @authors tag. See the
* copyright.txt file in the distribution for a full listing of all
* contributors.
*
* 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.mapstruct.ap.test.bugs._1482;
public interface ValueWrapper<T> {
T getValue();
}