Skip to content

Commit

Permalink
Fix SortedList.addAll(...) to be JLS/API compliant and provide unit t…
Browse files Browse the repository at this point in the history
…ests (Fixes #6034)

SortedList.addAll(...) should return a boolean value indicating if the Collection was modified is any way as a result of the call
  • Loading branch information
Luís Mendes committed Nov 12, 2024
1 parent ddf0233 commit 0dfe07c
Show file tree
Hide file tree
Showing 2 changed files with 156 additions and 2 deletions.
4 changes: 2 additions & 2 deletions src/main/java/spoon/support/util/SortedList.java
Original file line number Diff line number Diff line change
Expand Up @@ -42,9 +42,9 @@ public void add(int index, E element) {

@Override
public boolean addAll(Collection<? extends E> c) {
boolean ret = true;
boolean ret = false;
for (E e : c) {
ret &= add(e);
ret |= add(e);

Check warning on line 47 in src/main/java/spoon/support/util/SortedList.java

View workflow job for this annotation

GitHub Actions / code-quality qodana

Non-short-circuit boolean expression

Non-short-circuit boolean expression `ret |= add(e)`
}
return ret;
}
Expand Down
154 changes: 154 additions & 0 deletions src/test/java/spoon/support/util/SortedListTest.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,154 @@
package spoon.support.util;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.verify;

import java.util.ArrayList;
import java.util.Comparator;
import java.util.LinkedList;
import java.util.List;

import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.function.Executable;

import spoon.reflect.declaration.CtClass;
import spoon.reflect.declaration.CtElement;

public class SortedListTest {
public class BasicComparator implements Comparator<CtElement> {

private final List<CtElement> myOrdering;

public BasicComparator(List<CtElement> orderList) {
this.myOrdering = orderList;
}

@Override
public int compare(CtElement o1, CtElement o2) {
int idxFirst = - 1;
int idxSecond = -1;
int idx = 1;
for(CtElement clazz : myOrdering) {
if (o1 == o2 && o1 == clazz) {
return 0;
} else if (clazz == o1) {
if (idxFirst == -1) {
idxFirst = idx;
idx++;
} else {
throw new IllegalStateException();
}

} else if (clazz == o2) {
if (idxSecond == -1) {
idxSecond = idx;
idx++;
} else {
throw new IllegalStateException();
}
};
}

return idxFirst - idxSecond;
}

}

@Test
public void sortedListUsesComparatorTestPass() {
Comparator<CtClass<?>> cmp1 = mock();
SortedList<CtClass<?>> sl1 = new SortedList<>(cmp1);
CtClass<?> lc1 = mock();
CtClass<?> lc2 = mock();
assertTrue(sl1.add(lc1), "Element should have been inserted");
assertTrue(sl1.add(lc2), "Element should have been inserted");
verify(cmp1).compare(lc2, lc1);
}

@Test
public void verifyAddWithIndexThrowsException() {
Comparator<CtClass<?>> cmp1 = mock();
SortedList<CtClass<?>> sl1 = new SortedList<>(cmp1);
CtClass<?> lc1 = mock();
IllegalArgumentException ex = assertThrows(
IllegalArgumentException.class,
new Executable() {

@Override
public void execute() throws Throwable {
sl1.add(1, lc1);
}

},
"Add with index should trhow IllegalArgumentException");
assertEquals(
"cannot force a position with a sorted list that has its own ordering",
ex.getMessage(),
"Exception message does not match the expected");
}

@Test
public void verifyThatGetComparatorReturnsTheComparatorTestPass() {
Comparator<CtClass<?>> cmp1 = mock();
SortedList<CtClass<?>> sl1 = new SortedList<>(cmp1);
assertTrue(cmp1 == sl1.getComparator(), "Comparator does not match the expected");
}

@Test
public void verifyThatComparatorCanBeReplacedTestPass() {
Comparator<? super CtClass<?>> cmp1 = mock();
SortedList<CtClass<?>> sl1 = new SortedList<>(cmp1);
BasicComparator bc1 = new BasicComparator(null);
sl1.setComparator(bc1);
assertTrue(bc1 == sl1.getComparator(), "Comparator does not match the expected");
}

@Test
public void addAllInsertsAllElementsAndTheyAreOrderedTestPass() {
List<CtElement> orderedList = new ArrayList<>(3);
CtClass<?> lc1 = mock();
CtClass<?> lc2 = mock();
CtClass<?> lc3 = mock();
orderedList.add(lc1);
orderedList.add(lc2);
orderedList.add(lc3);

BasicComparator bc1 = new BasicComparator(orderedList);
SortedList<CtElement> sl1 = new SortedList<>(bc1);

List<CtElement> unorderedList = new ArrayList<>(3);
unorderedList.add(lc3);
unorderedList.add(lc1);
unorderedList.add(lc2);
assertTrue(sl1.addAll(unorderedList), "All elements should have been added");

//Verify ordering
assertTrue(sl1.get(0) == lc1,
"First element does not match the expected ordering");
assertTrue(sl1.get(1) == lc2,
"Second element does not match the expected ordering");
assertTrue(sl1.get(2) == lc3,
"Third element does not match the expected ordering");
}

@Test
public void addAllWithEmptyListShouldLeaveTheListUnchangedTestPass() {
List<CtElement> orderedList = new ArrayList<>(3);
CtClass<?> lc1 = mock();
CtClass<?> lc2 = mock();
CtClass<?> lc3 = mock();
orderedList.add(lc1);
orderedList.add(lc2);
orderedList.add(lc3);

BasicComparator bc1 = new BasicComparator(orderedList);
SortedList<CtElement> sl1 = new SortedList<>(bc1);

assertTrue(sl1.add(lc1), "List should have changed");
assertFalse(sl1.addAll(new LinkedList<>()), "List should not have changed");
}
}

0 comments on commit 0dfe07c

Please sign in to comment.