Skip to content

Commit

Permalink
Prioritize the default specialization (#22)
Browse files Browse the repository at this point in the history
* Prioritize the default specialization

See #18

This PR doesn't choose the default specialization by inspecting
parameter values. Instead it just takes the first specialization
arbitrarily.

* spotless

---------

Co-authored-by: Jesse Wilson <[email protected]>
  • Loading branch information
swankjesse and squarejesse authored Oct 16, 2024
1 parent ab8bb79 commit f5ba281
Show file tree
Hide file tree
Showing 4 changed files with 112 additions and 70 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ package app.cash.burst.gradle
import assertk.assertThat
import assertk.assertions.contains
import assertk.assertions.containsExactlyInAnyOrder
import assertk.assertions.isEmpty
import assertk.assertions.isEqualTo
import assertk.assertions.isFalse
import assertk.assertions.isTrue
Expand All @@ -44,21 +45,24 @@ class BurstGradlePluginTest {

assertThat(testSuite.testCases.map { it.name }).containsExactlyInAnyOrder(
"test[jvm]",
"test_Decaf_Oat[jvm]",
"test_Regular_Milk[jvm]",
"test_Regular_None[jvm]",
"test_Decaf_Milk[jvm]",
"test_Decaf_None[jvm]",
"test_Decaf_Oat[jvm]",
"test_Double_Milk[jvm]",
"test_Double_None[jvm]",
"test_Regular_Oat[jvm]",
"test_Double_Oat[jvm]",
"test_Regular_Milk[jvm]",
"test_Regular_None[jvm]",
"test_Regular_Oat[jvm]",
)

val originalTest = testSuite.testCases.single { it.name == "test[jvm]" }
assertThat(originalTest.skipped).isTrue()
assertThat(originalTest.skipped).isFalse()

val defaultSpecialization = testSuite.testCases.single { it.name == "test_Decaf_None[jvm]" }
assertThat(defaultSpecialization.skipped).isTrue()

val sampleSpecialization = testSuite.testCases.single { it.name == "test_Decaf_Oat[jvm]" }
val sampleSpecialization = testSuite.testCases.single { it.name == "test_Regular_Milk[jvm]" }
assertThat(sampleSpecialization.skipped).isFalse()
}

Expand All @@ -78,22 +82,25 @@ class BurstGradlePluginTest {

assertThat(testSuite.testCases.map { it.name }).containsExactlyInAnyOrder(
"test",
"test_Decaf_Oat",
"test_Regular_Milk",
"test_Regular_None",
"test_Decaf_Milk",
"test_Decaf_None",
"test_Decaf_Oat",
"test_Double_Milk",
"test_Double_None",
"test_Regular_Oat",
"test_Double_Oat",
"test_Regular_Milk",
"test_Regular_None",
"test_Regular_Oat",
)

val originalTest = testSuite.testCases.single { it.name == "test" }
assertThat(originalTest.skipped).isTrue()
assertThat(originalTest.skipped).isFalse()

val defaultSpecialization = testSuite.testCases.single { it.name == "test_Decaf_None" }
assertThat(defaultSpecialization.skipped).isTrue()

val sampleVariant = testSuite.testCases.single { it.name == "test_Decaf_Oat" }
assertThat(sampleVariant.skipped).isFalse()
val sampleSpecialization = testSuite.testCases.single { it.name == "test_Regular_Milk" }
assertThat(sampleSpecialization.skipped).isFalse()
}

@Test
Expand All @@ -110,16 +117,28 @@ class BurstGradlePluginTest {
val coffeeTest = readTestSuite(testResults.resolve("test/TEST-CoffeeTest.xml"))
val coffeeTestTest = coffeeTest.testCases.single()
assertThat(coffeeTestTest.name).isEqualTo("test")
assertThat(coffeeTestTest.skipped).isTrue()
assertThat(coffeeTest.systemOut).isEqualTo(
"""
|set up Decaf None
|running Decaf None
|
""".trimMargin(),
)

val sampleTest = readTestSuite(testResults.resolve("test/TEST-CoffeeTest_Decaf_None.xml"))
val defaultTest = readTestSuite(testResults.resolve("test/TEST-CoffeeTest_Decaf_None.xml"))
val defaultTestTest = defaultTest.testCases.single()
assertThat(defaultTestTest.name).isEqualTo("test")
assertThat(defaultTestTest.skipped).isTrue()
assertThat(defaultTest.systemOut).isEmpty()

val sampleTest = readTestSuite(testResults.resolve("test/TEST-CoffeeTest_Regular_Milk.xml"))
val sampleTestTest = sampleTest.testCases.single()
assertThat(sampleTestTest.name).isEqualTo("test")
assertThat(sampleTestTest.skipped).isFalse()
assertThat(sampleTest.systemOut).isEqualTo(
"""
|set up Decaf None
|running Decaf None
|set up Regular Milk
|running Regular Milk
|
""".trimMargin(),
)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,7 @@ import org.jetbrains.kotlin.compiler.plugin.ExperimentalCompilerApi
@OptIn(ExperimentalCompilerApi::class)
class BurstKotlinPluginTest {
@Test
fun happyPath() {
fun functionParameters() {
val result = compile(
sourceFile = SourceFile.kotlin(
"CoffeeTest.kt",
Expand Down Expand Up @@ -71,29 +71,27 @@ class BurstKotlinPluginTest {
assertThat(originalTest.isAnnotationPresent(Test::class.java)).isFalse()

// Burst adds a specialization for each combination of parameters.
val sampleFunction = testClass.getMethod("test_Decaf_None")
assertThat(sampleFunction.isAnnotationPresent(Test::class.java)).isTrue()
assertThat(sampleFunction.isAnnotationPresent(Ignore::class.java)).isFalse()
sampleFunction.invoke(adapterInstance)
val sampleSpecialization = testClass.getMethod("test_Regular_Milk")
assertThat(sampleSpecialization.isAnnotationPresent(Test::class.java)).isTrue()
assertThat(sampleSpecialization.isAnnotationPresent(Ignore::class.java)).isFalse()
sampleSpecialization.invoke(adapterInstance)
assertThat(log).containsExactly("running Regular Milk")
log.clear()

// The first specialization is also annotated `@Ignore`.
val firstSpecialization = testClass.getMethod("test_Decaf_None")
assertThat(firstSpecialization.isAnnotationPresent(Test::class.java)).isTrue()
assertThat(firstSpecialization.isAnnotationPresent(Ignore::class.java)).isTrue()
firstSpecialization.invoke(adapterInstance)
assertThat(log).containsExactly("running Decaf None")
log.clear()

// Burst adds a no-parameter function that calls each specialization in sequence.
// Burst adds a no-parameter function that calls the first specialization.
val noArgsTest = testClass.getMethod("test")
assertThat(noArgsTest.isAnnotationPresent(Test::class.java)).isTrue()
assertThat(noArgsTest.isAnnotationPresent(Ignore::class.java)).isTrue()
assertThat(noArgsTest.isAnnotationPresent(Ignore::class.java)).isFalse()
noArgsTest.invoke(adapterInstance)
assertThat(log).containsExactly(
"running Decaf None",
"running Decaf Milk",
"running Decaf Oat",
"running Regular None",
"running Regular Milk",
"running Regular Oat",
"running Double None",
"running Double Milk",
"running Double Oat",
)
assertThat(log).containsExactly("running Decaf None")
}

@Test
Expand Down Expand Up @@ -156,9 +154,9 @@ class BurstKotlinPluginTest {

val baseClass = result.classLoader.loadClass("CoffeeTest")

// Burst opens the class because it needs to subclass it. And it marks the entire class @Ignore.
// Burst opens the class because it needs to subclass it.
assertThat(Modifier.isFinal(baseClass.modifiers)).isFalse()
assertThat(baseClass.isAnnotationPresent(Ignore::class.java)).isTrue()
assertThat(baseClass.isAnnotationPresent(Ignore::class.java)).isFalse()

// Burst adds a no-args constructor that binds the first enum value.
val baseConstructor = baseClass.constructors.single { it.parameterCount == 0 }
Expand All @@ -176,18 +174,23 @@ class BurstKotlinPluginTest {
baseLog.clear()

// It generates a subclass for each specialization.
val sampleClass = result.classLoader.loadClass("CoffeeTest_Regular_Oat")
val sampleClass = result.classLoader.loadClass("CoffeeTest_Regular_Milk")
assertThat(sampleClass.isAnnotationPresent(Ignore::class.java)).isFalse()
val sampleConstructor = sampleClass.getConstructor()
val sampleInstance = sampleConstructor.newInstance()
val sampleLog = sampleClass.getMethod("getLog")
.invoke(sampleInstance) as MutableList<*>
sampleClass.getMethod("setUp").invoke(sampleInstance)
sampleClass.getMethod("test").invoke(sampleInstance)
assertThat(sampleLog).containsExactly(
"set up Regular Oat",
"running Regular Oat",
"set up Regular Milk",
"running Regular Milk",
)
sampleLog.clear()

// The default specialization is annotated `@Ignore`.
val defaultClass = result.classLoader.loadClass("CoffeeTest_Decaf_None")
assertThat(defaultClass.isAnnotationPresent(Ignore::class.java)).isTrue()
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@
package app.cash.burst.kotlin

import org.jetbrains.kotlin.backend.common.extensions.IrPluginContext
import org.jetbrains.kotlin.descriptors.DescriptorVisibilities
import org.jetbrains.kotlin.descriptors.Modality
import org.jetbrains.kotlin.ir.builders.declarations.addConstructor
import org.jetbrains.kotlin.ir.builders.declarations.buildClass
Expand Down Expand Up @@ -43,13 +44,12 @@ import org.jetbrains.kotlin.name.Name
* }
* ```
*
* This opens the class, adds `@Ignore`, and adds a default constructor that calls the first
* specialization:
* This opens the class, makes that constructor protected, and adds a default constructor that calls
* the first specialization:
*
* ```
* @Burst
* @Ignore
* open class CoffeeTest(
* open class CoffeeTest protected constructor(
* private val espresso: Espresso,
* private val dairy: Dairy,
* ) {
Expand All @@ -58,10 +58,11 @@ import org.jetbrains.kotlin.name.Name
* }
* ```
*
* And it generates a new test class for each specialization:
* And it generates a new test class for each specialization. The default specialization is also
* annotated `@Ignore`.
*
* ```
* class CoffeeTest_Decaf_None : CoffeeTest(Espresso.Decaf, Dairy.None)
* @Ignore class CoffeeTest_Decaf_None : CoffeeTest(Espresso.Decaf, Dairy.None)
* class CoffeeTest_Decaf_Milk : CoffeeTest(Espresso.Decaf, Dairy.Milk)
* class CoffeeTest_Decaf_Oat : CoffeeTest(Espresso.Decaf, Dairy.Oat)
* class CoffeeTest_Regular_None : CoffeeTest(Espresso.Regular, Dairy.None)
Expand Down Expand Up @@ -90,24 +91,33 @@ internal class ClassSpecializer(
}

val cartesianProduct = parameterArguments.cartesianProduct()
val defaultSpecialization = cartesianProduct.first()

// Add @Ignore and open the class
// TODO: don't double-add @Ignore
original.annotations += burstApis.ignoreClassSymbol.asAnnotation()
original.modality = Modality.OPEN
onlyConstructor.visibility = DescriptorVisibilities.PROTECTED

// Add a no-args constructor that calls the only constructor.
createNoArgsConstructor(onlyConstructor, cartesianProduct.first())
// Add a no-args constructor that calls the only constructor as the default specialization.
createNoArgsConstructor(
superConstructor = onlyConstructor,
arguments = defaultSpecialization,
)

// Add a subclass for each specialization.
cartesianProduct.map { arguments ->
createSpecialization(onlyConstructor, arguments)
createSpecialization(
superConstructor = onlyConstructor,
arguments = arguments,
isDefaultSpecialization = arguments == defaultSpecialization,
)
}
}

private fun createSpecialization(
superConstructor: IrConstructor,
arguments: List<Argument>,
isDefaultSpecialization: Boolean,
) {
val specialization = original.factory.buildClass {
initDefaults(original)
Expand All @@ -117,6 +127,10 @@ internal class ClassSpecializer(
createImplicitParameterDeclarationWithWrappedDescriptor()
}

if (isDefaultSpecialization) {
specialization.annotations += burstApis.ignoreClassSymbol.asAnnotation()
}

specialization.addConstructor {
initDefaults(original)
}.apply {
Expand Down
Loading

0 comments on commit f5ba281

Please sign in to comment.