@@ -17,27 +17,48 @@ package com.diffplug.selfie.junit5
17
17
18
18
import com.diffplug.selfie.RW
19
19
import com.diffplug.selfie.Snapshot
20
+ import java.nio.file.Files
21
+ import java.nio.file.Paths
20
22
import java.util.stream.Collectors
23
+ import kotlin.io.path.name
21
24
22
25
/* * 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
27
44
}
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 )"
29
48
}
30
49
/* * Represents the callstack above a given CallLocation. */
31
50
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
+ }
33
54
}
34
55
/* * Generates a CallLocation and the CallStack behind it. */
35
56
fun recordCall (): CallStack {
36
57
val calls =
37
58
StackWalker .getInstance().walk { frames ->
38
59
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) }
41
62
.collect(Collectors .toList())
42
63
}
43
64
return CallStack (calls.removeAt(0 ), calls)
@@ -53,7 +74,7 @@ internal open class WriteTracker<K : Comparable<K>, V> {
53
74
if (existing != null ) {
54
75
if (existing.snapshot != snapshot) {
55
76
throw org.opentest4j.AssertionFailedError (
56
- " Snapshot was set to multiple values: \n first time:${existing.callStack} \n\n this time:${call} " ,
77
+ " Snapshot was set to multiple values! \n first time: ${existing.callStack} \n this time: ${call} " ,
57
78
existing.snapshot,
58
79
snapshot)
59
80
} else if (RW .isWriteOnce) {
0 commit comments