Skip to content

Commit

Permalink
kotlinx-coroutines-debug: Fix split-package with the -core module
Browse files Browse the repository at this point in the history
- `kotlinx.coroutines.debug.internal` owned by coroutines-core
- `kotlinx.coroutines.debug` owned by coroutines-debug

(Which also reflects the nature of the debug module, which exposes _some_
public API from the .debug.internal package)

This requires moving:
- AgentPremain ->  .debug.internal
- ByteBuddyDynamicAttach -> .debug
- NoOpProbes.kt -> .debug
  • Loading branch information
sellmair committed Oct 10, 2024
1 parent 6c6df2b commit 59dc652
Show file tree
Hide file tree
Showing 16 changed files with 121 additions and 17 deletions.
2 changes: 2 additions & 0 deletions integration-testing/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
.kotlin
kotlin-js-store
49 changes: 48 additions & 1 deletion integration-testing/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -52,6 +52,7 @@ repositories {
java {
sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8
modularity.inferModulePath = true
}

dependencies {
Expand Down Expand Up @@ -117,6 +118,46 @@ sourceSets {
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutines_version"
}
}

debugDynamicAgentJpmsTest {
compileClasspath += sourceSets.test.runtimeClasspath
runtimeClasspath += sourceSets.test.runtimeClasspath

dependencies {
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutines_version"
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-debug:$coroutines_version"
}
}

tasks.compileDebugDynamicAgentJpmsTestKotlin.configure {
compilerOptions {
jvmTarget = org.jetbrains.kotlin.gradle.dsl.JvmTarget.JVM_17
}
}

tasks.compileDebugDynamicAgentJpmsTestJava.configure {
options.release.set(17)
}

debugDynamicAgentJpmsTest {
compileClasspath += sourceSets.test.runtimeClasspath
runtimeClasspath += sourceSets.test.runtimeClasspath

dependencies {
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutines_version"
implementation "org.jetbrains.kotlinx:kotlinx-coroutines-debug:$coroutines_version"
}
}

tasks.compileDebugDynamicAgentJpmsTestKotlin.configure {
compilerOptions {
jvmTarget = org.jetbrains.kotlin.gradle.dsl.JvmTarget.JVM_17
}
}

tasks.compileDebugDynamicAgentJpmsTestJava.configure {
options.release.set(17)
}
}

compileDebugAgentTestKotlin {
Expand Down Expand Up @@ -154,6 +195,12 @@ task debugDynamicAgentTest(type: Test) {
classpath = sourceSet.runtimeClasspath
}

task debugDynamicAgentJpmsTest(type: Test) {
def sourceSet = sourceSets.debugDynamicAgentJpmsTest
testClassesDirs = sourceSet.output.classesDirs
classpath = sourceSet.runtimeClasspath
}

task coreAgentTest(type: Test) {
def sourceSet = sourceSets.coreAgentTest
def coroutinesCoreJar = sourceSet.runtimeClasspath.filter {it.name == "kotlinx-coroutines-core-jvm-${coroutines_version}.jar" }.singleFile
Expand All @@ -167,7 +214,7 @@ compileTestKotlin {
}

check {
dependsOn([jvmCoreTest, debugDynamicAgentTest, mavenTest, debugAgentTest, coreAgentTest, 'smokeTest:build'])
dependsOn([jvmCoreTest, debugDynamicAgentTest, mavenTest, debugAgentTest, coreAgentTest, debugDynamicAgentJpmsTest, 'smokeTest:build'])
}
compileKotlin {
kotlinOptions {
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
module debug.dynamic.agent.jpms.test {
requires kotlin.stdlib;
requires kotlinx.coroutines.core;
requires kotlinx.coroutines.debug;
requires junit;
requires kotlin.test;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
@file:OptIn(ExperimentalCoroutinesApi::class)

import org.junit.*
import kotlinx.coroutines.*
import kotlinx.coroutines.debug.*
import org.junit.Ignore
import org.junit.Test
import java.io.*
import java.lang.IllegalStateException
import kotlin.test.*

class DynamicAttachDebugTest {

/**
* Using:
*
* jvmArgs("--add-exports=kotlinx.coroutines.debug/kotlinx.coroutines.repackaged.net.bytebuddy=com.sun.jna")
* jvmArgs("--add-exports=kotlinx.coroutines.debug/kotlinx.coroutines.repackaged.net.bytebuddy.agent=com.sun.jna")
*
*
* Caused by: java.lang.IllegalStateException: The Byte Buddy agent is not loaded or this method is not called via the system class loader
* at kotlinx.coroutines.debug/kotlinx.coroutines.repackaged.net.bytebuddy.agent.Installer.getInstrumentation(Installer.java:61)
* ... 54 more
*/
@Ignore("shaded byte-buddy does not work with JPMS")
@Test
fun testAgentDumpsCoroutines() =
DebugProbes.withDebugProbes {
runBlocking {
val baos = ByteArrayOutputStream()
DebugProbes.dumpCoroutines(PrintStream(baos))
// if the agent works, then dumps should contain something,
// at least the fact that this test is running.
Assert.assertTrue(baos.toString().contains("testAgentDumpsCoroutines"))
}
}

@Test()
fun testAgentIsNotInstalled() {
assertEquals(false, DebugProbes.isInstalled)
assertFailsWith<IllegalStateException> {
DebugProbes.dumpCoroutines(PrintStream(ByteArrayOutputStream()))
}
}

}
2 changes: 1 addition & 1 deletion kotlinx-coroutines-core/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -188,7 +188,7 @@ val allMetadataJar by tasks.getting(Jar::class) { setupManifest(this) }
fun setupManifest(jar: Jar) {
jar.manifest {
attributes(mapOf(
"Premain-Class" to "kotlinx.coroutines.debug.AgentPremain",
"Premain-Class" to "kotlinx.coroutines.debug.internal.AgentPremain",
"Can-Retransform-Classes" to "true",
))
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
volatile <fields>;
}

# These classes are only required by kotlinx.coroutines.debug.AgentPremain, which is only loaded when
# These classes are only required by kotlinx.coroutines.debug.internal.AgentPremain, which is only loaded when
# kotlinx-coroutines-core is used as a Java agent, so these are not needed in contexts where ProGuard is used.
-dontwarn java.lang.instrument.ClassFileTransformer
-dontwarn sun.misc.SignalHandler
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@
volatile <fields>;
}

# These classes are only required by kotlinx.coroutines.debug.AgentPremain, which is only loaded when
# These classes are only required by kotlinx.coroutines.debug.internal.AgentPremain, which is only loaded when
# kotlinx-coroutines-core is used as a Java agent, so these are not needed in contexts where ProGuard is used.
-dontwarn java.lang.instrument.ClassFileTransformer
-dontwarn sun.misc.SignalHandler
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
volatile <fields>;
}

# These classes are only required by kotlinx.coroutines.debug.AgentPremain, which is only loaded when
# These classes are only required by kotlinx.coroutines.debug.internal.AgentPremain, which is only loaded when
# kotlinx-coroutines-core is used as a Java agent, so these are not needed in contexts where ProGuard is used.
-dontwarn java.lang.instrument.ClassFileTransformer
-dontwarn sun.misc.SignalHandler
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,13 @@ package kotlinx.coroutines.debug.internal
/**
* Object used to differentiate between agent installed statically or dynamically.
* This is done in a separate object so [DebugProbesImpl] can check for static installation
* without having to depend on [kotlinx.coroutines.debug.AgentPremain], which is not compatible with Android.
* without having to depend on [AgentPremain], which is not compatible with Android.
* Otherwise, access to `AgentPremain.isInstalledStatically` triggers the load of its internal `ClassFileTransformer`
* that is not available on Android.
*
* Usage Note: Fleet (Reflection): FleetDebugProbes
* Usage Note: Android (Hard Coded, ignored for Leak Detection)
* Usage Note: IntelliJ (Suppress KotlinInternalInJava): CoroutineDumpState
*/
internal object AgentInstallationType {
internal var isInstalledStatically = false
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,6 @@
package kotlinx.coroutines.debug
package kotlinx.coroutines.debug.internal

import android.annotation.*
import kotlinx.coroutines.debug.internal.*
import org.codehaus.mojo.animal_sniffer.*
import sun.misc.*
import java.lang.instrument.*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ internal object DebugProbesImpl {

@Suppress("UNCHECKED_CAST")
private fun getDynamicAttach(): Function1<Boolean, Unit>? = runCatching {
val clz = Class.forName("kotlinx.coroutines.debug.internal.ByteBuddyDynamicAttach")
val clz = Class.forName("kotlinx.coroutines.debug.ByteBuddyDynamicAttach")
val ctor = clz.constructors[0]
ctor.newInstance() as Function1<Boolean, Unit>
}.getOrNull()
Expand Down
3 changes: 1 addition & 2 deletions kotlinx-coroutines-core/jvm/src/module-info.java
Original file line number Diff line number Diff line change
Expand Up @@ -5,13 +5,12 @@
requires transitive kotlin.stdlib;
requires kotlinx.atomicfu;

// these are used by kotlinx.coroutines.debug.AgentPremain
// these are used by kotlinx.coroutines.debug.internal.AgentPremain
requires static java.instrument; // contains java.lang.instrument.*
requires static jdk.unsupported; // contains sun.misc.Signal

exports kotlinx.coroutines;
exports kotlinx.coroutines.channels;
exports kotlinx.coroutines.debug;
exports kotlinx.coroutines.debug.internal;
exports kotlinx.coroutines.flow;
exports kotlinx.coroutines.flow.internal;
Expand Down
4 changes: 2 additions & 2 deletions kotlinx-coroutines-debug/build.gradle.kts
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ val shadowJar by tasks.existing(ShadowJar::class) {
manifest {
attributes(
mapOf(
"Premain-Class" to "kotlinx.coroutines.debug.AgentPremain",
"Premain-Class" to "kotlinx.coroutines.debug.internal.AgentPremain",
"Can-Redefine-Classes" to "true",
"Multi-Release" to "true"
)
Expand Down Expand Up @@ -104,7 +104,7 @@ kover {
filters {
excludes {
// Never used, safety mechanism
classes("kotlinx.coroutines.debug.internal.NoOpProbesKt")
classes("kotlinx.coroutines.debug.NoOpProbesKt")
}
}
}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
@file:Suppress("unused")
package kotlinx.coroutines.debug.internal
package kotlinx.coroutines.debug

import net.bytebuddy.*
import net.bytebuddy.agent.*
Expand Down Expand Up @@ -28,7 +28,7 @@ internal class ByteBuddyDynamicAttach : Function1<Boolean, Unit> {

private fun detach() {
val cl = Class.forName("kotlin.coroutines.jvm.internal.DebugProbesKt")
val cl2 = Class.forName("kotlinx.coroutines.debug.internal.NoOpProbesKt")
val cl2 = Class.forName("kotlinx.coroutines.debug.NoOpProbesKt")
ByteBuddy()
.redefine(cl2)
.name(cl.name)
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
@file:Suppress("unused", "UNUSED_PARAMETER")

package kotlinx.coroutines.debug.internal
package kotlinx.coroutines.debug

import kotlin.coroutines.*

Expand Down
2 changes: 1 addition & 1 deletion kotlinx-coroutines-debug/src/module-info.java
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
requires org.junit.jupiter.api;
requires org.junit.platform.commons;

// exports kotlinx.coroutines.debug; // already exported by kotlinx.coroutines.core
exports kotlinx.coroutines.debug;
exports kotlinx.coroutines.debug.junit4;
exports kotlinx.coroutines.debug.junit5;
}

0 comments on commit 59dc652

Please sign in to comment.