-
-
Notifications
You must be signed in to change notification settings - Fork 357
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
review: fix: stackoverflow on AbstractTypingContext #1817
Changes from 2 commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -24,6 +24,7 @@ | |
import spoon.reflect.declaration.CtType; | ||
import spoon.reflect.declaration.ModifierKind; | ||
import spoon.reflect.factory.Factory; | ||
import spoon.reflect.visitor.filter.NamedElementFilter; | ||
import spoon.test.delete.testclasses.Adobada; | ||
import spoon.test.method.testclasses.Tacos; | ||
|
||
|
@@ -85,4 +86,18 @@ public void testGetAllMethods() throws Exception { | |
assertEquals(1, methods.stream().filter(method -> "foo".equals(method.getSimpleName())).count()); | ||
} | ||
|
||
@Test | ||
public void testGetAllMethodsAdaptingType() throws Exception { | ||
// contract: AbstractTypingContext should not enter in recursive calls when resolving autoreferenced bounding type | ||
// such as T extends Comparable<? super T> | ||
Launcher l = new Launcher(); | ||
l.getEnvironment().setNoClasspath(true); | ||
l.addInputResource("src/test/resources/noclasspath/spring/PropertyComparator.java"); | ||
l.buildModel(); | ||
|
||
CtType<?> propertyComparator = l.getModel().getElements(new NamedElementFilter<CtType>(CtType.class, "PropertyComparator")).get(0); | ||
CtMethod method = propertyComparator.getMethodsByName("compare").get(0); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Which call failed on StackOverflowError? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. getMethodsByName There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. So why did you removed this method call from the test?? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. :D sorry I answer yesterday evening and actually I was wrong: it's getAllMethods which led to the bug. My test wasn't failing after all yesterday. |
||
assertEquals("compare(T,T)", method.getSignature()); | ||
} | ||
|
||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,156 @@ | ||
/* | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Could you delete as much as possible from this file, so there remains only the code needed to reproduce error? |
||
* Copyright 2002-2017 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.beans.support; | ||
|
||
import java.util.Arrays; | ||
import java.util.Collections; | ||
import java.util.Comparator; | ||
import java.util.List; | ||
|
||
import org.apache.commons.logging.Log; | ||
import org.apache.commons.logging.LogFactory; | ||
|
||
import org.springframework.beans.BeanWrapperImpl; | ||
import org.springframework.beans.BeansException; | ||
import org.springframework.lang.Nullable; | ||
import org.springframework.util.StringUtils; | ||
|
||
/** | ||
* PropertyComparator performs a comparison of two beans, | ||
* evaluating the specified bean property via a BeanWrapper. | ||
* | ||
* @author Juergen Hoeller | ||
* @author Jean-Pierre Pawlak | ||
* @since 19.05.2003 | ||
* @see org.springframework.beans.BeanWrapper | ||
*/ | ||
public class PropertyComparator<T> implements Comparator<T> { | ||
|
||
protected final Log logger = LogFactory.getLog(getClass()); | ||
|
||
private final SortDefinition sortDefinition; | ||
|
||
private final BeanWrapperImpl beanWrapper = new BeanWrapperImpl(false); | ||
|
||
|
||
/** | ||
* Create a new PropertyComparator for the given SortDefinition. | ||
* @see MutableSortDefinition | ||
*/ | ||
public PropertyComparator(SortDefinition sortDefinition) { | ||
this.sortDefinition = sortDefinition; | ||
} | ||
|
||
/** | ||
* Create a PropertyComparator for the given settings. | ||
* @param property the property to compare | ||
* @param ignoreCase whether upper and lower case in String values should be ignored | ||
* @param ascending whether to sort ascending (true) or descending (false) | ||
*/ | ||
public PropertyComparator(String property, boolean ignoreCase, boolean ascending) { | ||
this.sortDefinition = new MutableSortDefinition(property, ignoreCase, ascending); | ||
} | ||
|
||
/** | ||
* Return the SortDefinition that this comparator uses. | ||
*/ | ||
public final SortDefinition getSortDefinition() { | ||
return this.sortDefinition; | ||
} | ||
|
||
|
||
@Override | ||
@SuppressWarnings("unchecked") | ||
public int compare(T o1, T o2) { | ||
Object v1 = getPropertyValue(o1); | ||
Object v2 = getPropertyValue(o2); | ||
if (this.sortDefinition.isIgnoreCase() && (v1 instanceof String) && (v2 instanceof String)) { | ||
v1 = ((String) v1).toLowerCase(); | ||
v2 = ((String) v2).toLowerCase(); | ||
} | ||
|
||
int result; | ||
|
||
// Put an object with null property at the end of the sort result. | ||
try { | ||
if (v1 != null) { | ||
result = (v2 != null ? ((Comparable<Object>) v1).compareTo(v2) : -1); | ||
} | ||
else { | ||
result = (v2 != null ? 1 : 0); | ||
} | ||
} | ||
catch (RuntimeException ex) { | ||
if (logger.isWarnEnabled()) { | ||
logger.warn("Could not sort objects [" + o1 + "] and [" + o2 + "]", ex); | ||
} | ||
return 0; | ||
} | ||
|
||
return (this.sortDefinition.isAscending() ? result : -result); | ||
} | ||
|
||
/** | ||
* Get the SortDefinition's property value for the given object. | ||
* @param obj the object to get the property value for | ||
* @return the property value | ||
*/ | ||
@Nullable | ||
private Object getPropertyValue(Object obj) { | ||
// If a nested property cannot be read, simply return null | ||
// (similar to JSTL EL). If the property doesn't exist in the | ||
// first place, let the exception through. | ||
try { | ||
this.beanWrapper.setWrappedInstance(obj); | ||
return this.beanWrapper.getPropertyValue(this.sortDefinition.getProperty()); | ||
} | ||
catch (BeansException ex) { | ||
logger.info("PropertyComparator could not access property - treating as null for sorting", ex); | ||
return null; | ||
} | ||
} | ||
|
||
|
||
/** | ||
* Sort the given List according to the given sort definition. | ||
* <p>Note: Contained objects have to provide the given property | ||
* in the form of a bean property, i.e. a getXXX method. | ||
* @param source the input List | ||
* @param sortDefinition the parameters to sort by | ||
* @throws java.lang.IllegalArgumentException in case of a missing propertyName | ||
*/ | ||
public static void sort(List<?> source, SortDefinition sortDefinition) throws BeansException { | ||
if (StringUtils.hasText(sortDefinition.getProperty())) { | ||
Collections.sort(source, new PropertyComparator<>(sortDefinition)); | ||
} | ||
} | ||
|
||
/** | ||
* Sort the given source according to the given sort definition. | ||
* <p>Note: Contained objects have to provide the given property | ||
* in the form of a bean property, i.e. a getXXX method. | ||
* @param source input source | ||
* @param sortDefinition the parameters to sort by | ||
* @throws java.lang.IllegalArgumentException in case of a missing propertyName | ||
*/ | ||
public static void sort(Object[] source, SortDefinition sortDefinition) throws BeansException { | ||
if (StringUtils.hasText(sortDefinition.getProperty())) { | ||
Arrays.sort(source, new PropertyComparator<>(sortDefinition)); | ||
} | ||
} | ||
|
||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
May this happen too?
T extends Comparable<? super List<? extends T>>
I guess it will end wit stackoverflow too.
Anyway your solution is good fast fix. My example is quite wild and it will need some time until somebody come to such problem again.