Skip to content

Commit 7de1aa4

Browse files
authored
Improve callstack capture and IDE-compatible printing (#46)
2 parents db1bec2 + 6931934 commit 7de1aa4

File tree

2 files changed

+32
-10
lines changed

2 files changed

+32
-10
lines changed

selfie-runner-junit5/src/main/kotlin/com/diffplug/selfie/junit5/WriteTracker.kt

+30-9
Original file line numberDiff line numberDiff line change
@@ -17,27 +17,48 @@ package com.diffplug.selfie.junit5
1717

1818
import com.diffplug.selfie.RW
1919
import com.diffplug.selfie.Snapshot
20+
import java.nio.file.Files
21+
import java.nio.file.Paths
2022
import java.util.stream.Collectors
23+
import kotlin.io.path.name
2124

2225
/** Represents the line at which user code called into Selfie. */
23-
data class CallLocation(val subpath: String, val line: Int) : Comparable<CallLocation> {
24-
override fun compareTo(other: CallLocation): Int {
25-
val subpathCompare = subpath.compareTo(other.subpath)
26-
return if (subpathCompare != 0) subpathCompare else line.compareTo(other.line)
26+
data class CallLocation(val clazz: String, val method: String, val file: String?, val line: Int) :
27+
Comparable<CallLocation> {
28+
override fun compareTo(other: CallLocation): Int =
29+
compareValuesBy(this, other, { it.clazz }, { it.method }, { it.file }, { it.line })
30+
31+
/**
32+
* If the runtime didn't give us the filename, guess it from the class, and try to find the source
33+
* file by walking the CWD. If we don't find it, report it as a `.class` file.
34+
*/
35+
private fun findFileIfAbsent(): String {
36+
if (file != null) {
37+
return file
38+
}
39+
val fileWithoutExtension = clazz.substringAfterLast('.').substringBefore('$')
40+
val likelyExtensions = listOf("kt", "java", "scala", "groovy", "clj", "cljc")
41+
val filenames = likelyExtensions.map { "$fileWithoutExtension.$it" }.toSet()
42+
val firstPath = Files.walk(Paths.get("")).use { it.filter { it.name in filenames }.findFirst() }
43+
return if (firstPath.isEmpty) "${clazz.substringAfterLast('.')}.class" else firstPath.get().name
2744
}
28-
override fun toString(): String = "$subpath:$line"
45+
46+
/** A `toString` which an IDE will render as a clickable link. */
47+
override fun toString(): String = "$clazz.$method(${findFileIfAbsent()}:$line)"
2948
}
3049
/** Represents the callstack above a given CallLocation. */
3150
class CallStack(val location: CallLocation, val restOfStack: List<CallLocation>) {
32-
override fun toString(): String = "$location"
51+
override fun toString(): String {
52+
return location.toString()
53+
}
3354
}
3455
/** Generates a CallLocation and the CallStack behind it. */
3556
fun recordCall(): CallStack {
3657
val calls =
3758
StackWalker.getInstance().walk { frames ->
3859
frames
39-
.skip(1)
40-
.map { CallLocation(it.className.replace('.', '/') + ".kt", it.lineNumber) }
60+
.dropWhile { it.className.startsWith("com.diffplug.selfie") }
61+
.map { CallLocation(it.className, it.methodName, it.fileName, it.lineNumber) }
4162
.collect(Collectors.toList())
4263
}
4364
return CallStack(calls.removeAt(0), calls)
@@ -53,7 +74,7 @@ internal open class WriteTracker<K : Comparable<K>, V> {
5374
if (existing != null) {
5475
if (existing.snapshot != snapshot) {
5576
throw org.opentest4j.AssertionFailedError(
56-
"Snapshot was set to multiple values:\nfirst time:${existing.callStack}\n\nthis time:${call}",
77+
"Snapshot was set to multiple values!\n first time: ${existing.callStack}\n this time: ${call}",
5778
existing.snapshot,
5879
snapshot)
5980
} else if (RW.isWriteOnce) {

selfie-runner-junit5/src/test/kotlin/testpkg/RecordCallTest.kt

+2-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,8 @@ class RecordCallTest {
2424
@Test
2525
fun testRecordCall() {
2626
val stack = recordCall()
27-
stack.location.toString() shouldBe "testpkg/RecordCallTest.kt:26"
27+
// shows as clickable link in IDE
28+
stack.location.toString() shouldBe "testpkg.RecordCallTest.testRecordCall(RecordCallTest.kt:26)"
2829
stack.restOfStack.size shouldBeGreaterThan 0
2930
}
3031
}

0 commit comments

Comments
 (0)