我正在尝试编写一个简单的程序来遍历从给定方法开始的所有引用代码,使用scalameta
.
我能够跟踪调用,但无法解析方法引用。
分析/src/main/scala/codelab/FindMe.scala
package codelab
object FindMe {
def main(args: Array[String]): Unit = {
val x = someRecognizeableName(1, 2)
val y = List(1, 2, 3)
y.foldLeft(0)(someRecognizeableName)
}
def someRecognizeableName(a: Int, b: Int): Int = a + b
}
生成并加载语义数据库FindMe.scala
并检查方法的用法someRecognizeableName
。
我可以看到db.names
列表中的第一个调用:
[87..108): someRecognizeableName => _root_.codelab.FindMe.someRecognizeableName(Int,Int).
但是,第二个,当我不调用该方法时,只需传递引用显示如下:
[159..180): someRecognizeableName => local2_src_main_scala_codelab_FindMe_scala
因此,当我尝试从 开始跟踪引用时,在第二种情况下main
,我没有得到引用的完全限定名称。someRecognizeableName
问题:有没有办法从语义数据库中为该引用获取一个完全限定的名称?
重现上述内容的完整来源
运行说明:
analyzeme $ sbt compile
analyzer $ sbt "run ../analyzeme"
分析/src/main/scala/codelab/FindMe.scala
package codelab
object FindMe {
def main(args: Array[String]): Unit = {
val x = someRecognizeableName(1, 2)
val y = List(1, 2, 3)
y.foldLeft(0)(someRecognizeableName)
}
def someRecognizeableName(a: Int, b: Int): Int = a + b
}
分析器/src/main/scala/Main.scala
import org.langmeta.io.{Classpath, Sourcepath}
import scala.meta._
object Main {
def main(args: Array[String]): Unit = {
println(s"Loading from [${ args(0) }]")
println()
val cp = Classpath(s"${ args(0) }/target/scala-2.12/classes")
val sp = Sourcepath(s"${ args(0) }/src/main/scala")
val db = Database.load(cp, sp)
println("* names:")
db.names foreach println
println()
println("* symbols:")
db.symbols foreach println
println()
println("* synthetics:")
db.synthetics foreach println
println()
println("* messages:")
db.messages foreach println
println()
}
}
分析/构建.sbt
name := "analyzee"
version := "0.1"
scalaVersion := "2.12.4"
addCompilerPlugin("org.scalameta" % "semanticdb-scalac" % "3.4.0" cross CrossVersion.full)
scalacOptions += "-Yrangepos"
分析器/build.sbt
name := "analyzer"
version := "0.1"
scalaVersion := "2.12.4"
libraryDependencies += "org.scalameta" %% "scalameta" % "3.4.0"
libraryDependencies += "org.scalameta" %% "contrib" % "3.4.0"