org.scalameta.explore.Macros.scala Maven / Gradle / Ivy
package org.scalameta
package explore
import org.scalameta.internal.MacroHelpers
import scala.collection.mutable.Set
import scala.reflect.macros.whitebox.Context
class ExploreMacros(val c: Context) extends MacroHelpers {
import c.universe._
lazy val m = c.mirror
private implicit class XtensionExploreSymbol(sym: Symbol) {
def isIrrelevant: Boolean = {
def artefact = sym.name.decodedName.toString.contains("$")
def trivial = sym.owner == symbolOf[Any] || sym.owner == symbolOf[Object]
def arbitrary = {
val banned = List(
"scala.collection.generic",
"scala.Enumeration",
"scala.math",
"scala.Int",
"scala.meta.inline.Api"
)
banned.exists(prefix => sym.fullName.startsWith(prefix))
}
def aliases = sym.fullName.contains(".Aliases.")
def contrib = sym.fullName.contains(".contrib.") || sym.fullName.endsWith(".contrib")
def interactive = sym.fullName.contains(".interactive.") ||
sym.fullName.endsWith(".interactive")
def testkit = sym.fullName.contains(".testkit.") || sym.fullName.endsWith(".testkit")
def tests = sym.fullName.contains(".tests.") || sym.fullName.endsWith(".tests")
def internal = sym.fullName.contains(".internal.") ||
(sym.fullName.endsWith(".internal") && !sym.fullName.endsWith(".meta.internal"))
def invisible = !sym.isPublic
def inexistent = !sym.asInstanceOf[scala.reflect.internal.SymbolTable#Symbol].exists // NOTE: wtf
artefact || trivial || arbitrary || aliases || contrib || interactive || testkit || tests ||
internal || invisible || inexistent
}
def isRelevant: Boolean = !isIrrelevant
def isPkgObject: Boolean = sym.owner.isPackage && sym.isModule && sym.name == termNames.PACKAGE
def isImplicitClass: Boolean = sym.isClass && sym.isImplicit
def signatureIn(owner: Symbol): String = {
def paramss(m: Symbol, wrapFirst: Boolean): String = {
val pss = m.asMethod.paramLists
val s_ps = pss.zipWithIndex.map { case (ps, i) =>
val s_p = ps.map(_.info).mkString(", ")
val designator = if (ps.exists(_.isImplicit)) "implicit " else ""
val prefix = if (i != 0 || wrapFirst) "(" else ""
val suffix = if (i != 0 || wrapFirst) ")" else ""
prefix + designator + s_p + suffix
}
s_ps.mkString("")
}
val prefix =
if (sym.owner.isImplicit) {
val ctor = sym.owner.info.members
.find(sym => sym.isMethod && sym.asMethod.isPrimaryConstructor).get
"* " + paramss(ctor, wrapFirst = false)
} else scala.reflect.NameTransformer.decode(owner.fullName)
val main = sym.name.decodedName.toString
val suffix =
if (sym.isConstructor || sym.isMacro || sym.isMethod) paramss(sym, wrapFirst = true) +
": " + sym.info.finalResultType
else ""
prefix + "." + main + suffix
}
}
private def staticClassesObjectsAndVals(
pkg: Symbol,
onlyImmediatelyAccessible: Boolean
): List[Symbol] = {
val visited = Set[Symbol]()
def loop(parent: Symbol): Unit = {
def mark(sym: Symbol): Boolean = {
val interesting = sym.isRelevant && !visited(sym)
if (interesting) visited += sym
interesting
}
def markAndRecur(sym: Symbol): Unit = if (mark(sym)) loop(sym)
if (parent == NoSymbol) return
parent.info.members.toList.foreach { sym =>
def failUnsupported(sym: Symbol): Nothing = c
.abort(c.enclosingPosition, s"unsupported symbol $sym: ${sym.fullName}")
if (sym.name.decodedName.toString.contains("$")) {
// ignore: not worth processing
} else if (sym.isPackage) {
if (!onlyImmediatelyAccessible) markAndRecur(sym)
visited += sym
} else if (sym.isModule)
// don't check onlyImmediatelyAccessible
// because we expect many members of modules to be used via full fullNames
markAndRecur(sym)
else if (sym.isClass)
// don't recur into class members
mark(sym)
else if (sym.isMethod) {
if (sym.asMethod.isGetter) {
val isInPackageClass = sym.owner.isModuleClass && sym.owner.name == typeNames.PACKAGE
if (isInPackageClass)
// doesn't make sense to recur into package class methods
mark(sym)
else {
// handle term aliases
val target = sym.info.finalResultType.typeSymbol
if (target.isModuleClass) loop(target.asClass.module) else ()
}
}
} else if (sym.isType)
// handle type aliases
// doesn't make sense to recur into type aliases
mark(sym.info.typeSymbol)
else if (sym.isTerm)
if (sym.asTerm.getter != NoSymbol || sym.isPrivateThis) {
// ignore: processed in sym.isMethod
} else failUnsupported(sym)
else failUnsupported(sym)
}
}
assert(pkg.isPackage)
loop(pkg)
var result = visited.toList
result = result.filter(_.isRelevant)
result
}
private def staticsImpl(packageName: Tree, onlyImmediatelyAccessible: Boolean): Tree = {
val Literal(Constant(s_packageName: String)) = packageName
val statics =
staticClassesObjectsAndVals(m.staticPackage(s_packageName), onlyImmediatelyAccessible)
val nonImplicitClassStatics = statics.filter(!_.isImplicitClass)
val (pkgObjects, nonPkgObjectStatics) = nonImplicitClassStatics.partition(_.isPkgObject)
val allowedPkgObjects = pkgObjects
.filter(sym => sym.owner.fullName == s_packageName || !onlyImmediatelyAccessible)
val pkgObjectMethods = allowedPkgObjects.flatMap(_.info.members.toList.collect {
case sym: MethodSymbol
if !sym.isAccessor && !sym.isImplicit && !sym.isConstructor && sym.isRelevant => sym
})
// NOTE: We filtered out package objects and implicit classes (hopefully, IDEs and autocompletes will ignore those),
// and then we added methods from package objects (because `import scala.meta._` will bring those in).
val effectiveStatics = nonPkgObjectStatics ++ pkgObjectMethods
val fullNames = effectiveStatics.map(sym => scala.reflect.NameTransformer.decode(sym.fullName))
q"${fullNames.distinct.sorted}"
}
def wildcardImportStaticsImpl(packageName: Tree): Tree =
staticsImpl(packageName, onlyImmediatelyAccessible = true)
def allStaticsImpl(packageName: Tree): Tree =
staticsImpl(packageName, onlyImmediatelyAccessible = false)
def extensionSurfaceImpl(packageName: Tree): Tree = {
val Literal(Constant(s_packageName: String)) = packageName
val statics =
staticClassesObjectsAndVals(m.staticPackage(s_packageName), onlyImmediatelyAccessible = false)
val extensionSurface = {
val implicitClasses = statics.filter(_.isImplicitClass)
val result = implicitClasses.flatMap { cls =>
val extensionMethods = cls.info.decls
.collect { case sym if sym.isMethod && !sym.isConstructor => sym.asMethod }
extensionMethods.filter(_.isRelevant)
}
result.map(sym => (NoSymbol, sym))
}
val s_surface = extensionSurface.collect { case (owner, sym) => sym.signatureIn(owner) }
q"${s_surface.distinct.sorted}"
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy