scala.tasty.reflect.SourceCodePrinter.scala Maven / Gradle / Ivy
The newest version!
package scala.tasty
package reflect
import scala.annotation.switch
import scala.quoted.show.SyntaxHighlight
/** Printer for fully elaborated representation of the source code */
class SourceCodePrinter[R <: Reflection & Singleton](val reflect: R)(syntaxHighlight: SyntaxHighlight) extends Printer[R] {
import reflect._
import syntaxHighlight._
def showTree(tree: Tree): String =
(new Buffer).printTree(tree).result()
def showType(tpe: Type): String =
(new Buffer).printType(tpe)(using None).result()
def showConstant(const: Constant): String =
(new Buffer).printConstant(const).result()
def showSymbol(symbol: Symbol): String =
symbol.fullName
def showFlags(flags: Flags): String = {
val flagList = List.newBuilder[String]
if (flags.is(Flags.Abstract)) flagList += "abstract"
if (flags.is(Flags.Artifact)) flagList += "artifact"
if (flags.is(Flags.Case)) flagList += "case"
if (flags.is(Flags.CaseAccessor)) flagList += "caseAccessor"
if (flags.is(Flags.Contravariant)) flagList += "contravariant"
if (flags.is(Flags.Covariant)) flagList += "covariant"
if (flags.is(Flags.Enum)) flagList += "enum"
if (flags.is(Flags.Erased)) flagList += "erased"
if (flags.is(Flags.ExtensionMethod)) flagList += "extension"
if (flags.is(Flags.FieldAccessor)) flagList += "accessor"
if (flags.is(Flags.Final)) flagList += "final"
if (flags.is(Flags.HasDefault)) flagList += "hasDefault"
if (flags.is(Flags.Implicit)) flagList += "implicit"
if (flags.is(Flags.Inline)) flagList += "inline"
if (flags.is(Flags.JavaDefined)) flagList += "javaDefined"
if (flags.is(Flags.Lazy)) flagList += "lazy"
if (flags.is(Flags.Local)) flagList += "local"
if (flags.is(Flags.Macro)) flagList += "macro"
if (flags.is(Flags.ModuleClass)) flagList += "moduleClass"
if (flags.is(Flags.Mutable)) flagList += "mutable"
if (flags.is(Flags.Object)) flagList += "object"
if (flags.is(Flags.Override)) flagList += "override"
if (flags.is(Flags.Package)) flagList += "package"
if (flags.is(Flags.Param)) flagList += "param"
if (flags.is(Flags.ParamAccessor)) flagList += "paramAccessor"
if (flags.is(Flags.Private)) flagList += "private"
if (flags.is(Flags.PrivateLocal)) flagList += "private[this]"
if (flags.is(Flags.Protected)) flagList += "protected"
if (flags.is(Flags.Scala2x)) flagList += "scala2x"
if (flags.is(Flags.Sealed)) flagList += "sealed"
if (flags.is(Flags.StableRealizable)) flagList += "stableRealizable"
if (flags.is(Flags.Static)) flagList += "javaStatic"
if (flags.is(Flags.Synthetic)) flagList += "synthetic"
if (flags.is(Flags.Trait)) flagList += "trait"
flagList.result().mkString("/*", " ", "*/")
}
private class Buffer {
private[this] val sb: StringBuilder = new StringBuilder
private[this] var indent: Int = 0
def indented(printIndented: => Unit): Unit = {
indent += 1
printIndented
indent -= 1
}
def inParens(body: => Unit): Buffer = {
this += "("
body
this += ")"
}
def inSquare(body: => Unit): Buffer = {
this += "["
body
this += "]"
}
def inBlock(body: => Unit): Buffer = {
this += " {"
indented {
this += lineBreak()
body
}
this += lineBreak() += "}"
}
def result(): String = sb.result()
def lineBreak(): String = "\n" + (" " * indent)
def doubleLineBreak(): String = "\n\n" + (" " * indent)
def printTree(tree: Tree)(using elideThis: Option[Symbol] = None): Buffer = tree match {
case PackageObject(body)=>
printTree(body) // Print package object
case PackageClause(Ident(name), (inner @ PackageClause(_, _)) :: Nil) if name != "" && PackageObject.unapply(inner).isEmpty =>
// print inner package as `package outer.inner { ... }`
printTree(inner)
case tree @ PackageClause(name, stats) =>
val stats1 = stats.collect {
case stat: PackageClause => stat
case stat: Definition if !(stat.symbol.flags.is(Flags.Object) && stat.symbol.flags.is(Flags.Lazy)) => stat
case stat @ Import(_, _) => stat
}
name match {
case Ident("") =>
printTrees(stats1, lineBreak())
case _ =>
this += "package "
printType(name.tpe)
inBlock(printTrees(stats1, lineBreak()))
}
case Import(expr, selectors) =>
this += "import "
printTree(expr)
this += "."
printImportSelectors(selectors)
case cdef @ ClassDef(name, DefDef(_, targs, argss, _, _), parents, derived, self, stats) =>
printDefAnnotations(cdef)
val flags = cdef.symbol.flags
if (flags.is(Flags.Implicit)) this += highlightKeyword("implicit ")
if (flags.is(Flags.Sealed)) this += highlightKeyword("sealed ")
if (flags.is(Flags.Final) && !flags.is(Flags.Object)) this += highlightKeyword("final ")
if (flags.is(Flags.Case)) this += highlightKeyword("case ")
if (name == "package$") {
this += highlightKeyword("package object ") += highlightTypeDef(cdef.symbol.owner.name.stripSuffix("$"))
}
else if (flags.is(Flags.Object)) this += highlightKeyword("object ") += highlightTypeDef(name.stripSuffix("$"))
else if (flags.is(Flags.Trait)) this += highlightKeyword("trait ") += highlightTypeDef(name)
else if (flags.is(Flags.Abstract)) this += highlightKeyword("abstract class ") += highlightTypeDef(name)
else this += highlightKeyword("class ") += highlightTypeDef(name)
val typeParams = stats.collect { case targ: TypeDef => targ }.filter(_.symbol.isTypeParam).zip(targs)
if (!flags.is(Flags.Object)) {
printTargsDefs(typeParams)
val it = argss.iterator
while (it.hasNext)
printArgsDefs(it.next())
}
val parents1 = parents.filter {
case Apply(Select(New(tpt), _), _) => tpt.tpe.typeSymbol != Symbol.requiredClass("java.lang.Object")
case TypeSelect(Select(Ident("_root_"), "scala"), "Product") => false
case TypeSelect(Select(Ident("_root_"), "scala"), "Serializable") => false
case _ => true
}
if (parents1.nonEmpty)
this += highlightKeyword(" extends ")
def printParent(parent: Tree /* Term | TypeTree */, needEmptyParens: Boolean = false): Unit = parent match {
case parent: TypeTree =>
printTypeTree(parent)(using Some(cdef.symbol))
case TypeApply(fun, targs) =>
printParent(fun)
case Apply(fun@Apply(_,_), args) =>
printParent(fun, true)
if (!args.isEmpty || needEmptyParens)
inParens(printTrees(args, ", ")(using Some(cdef.symbol)))
case Apply(fun, args) =>
printParent(fun)
if (!args.isEmpty || needEmptyParens)
inParens(printTrees(args, ", ")(using Some(cdef.symbol)))
case Select(newTree: New, _) =>
printType(newTree.tpe)(using Some(cdef.symbol))
case parent: Term =>
throw new MatchError(parent.showExtractors)
}
def printSeparated(list: List[Tree /* Term | TypeTree */]): Unit = list match {
case Nil =>
case x :: Nil => printParent(x)
case x :: xs =>
printParent(x)
this += highlightKeyword(" with ")
printSeparated(xs)
}
printSeparated(parents1)
if (derived.nonEmpty) {
this += highlightKeyword(" derives ")
printTypeTrees(derived, ", ")
}
def keepDefinition(d: Definition): Boolean = {
val flags = d.symbol.flags
def isUndecompilableCaseClassMethod: Boolean = {
// Currently the compiler does not allow overriding some of the methods generated for case classes
d.symbol.flags.is(Flags.Synthetic) &&
(d match {
case DefDef("apply" | "unapply" | "writeReplace", _, _, _, _) if d.symbol.owner.flags.is(Flags.Object) => true
case DefDef(n, _, _, _, _) if d.symbol.owner.flags.is(Flags.Case) =>
n == "copy" ||
n.matches("copy\\$default\\$[1-9][0-9]*") || // default parameters for the copy method
n.matches("_[1-9][0-9]*") || // Getters from Product
n == "productElementName"
case _ => false
})
}
def isInnerModuleObject = d.symbol.flags.is(Flags.Lazy) && d.symbol.flags.is(Flags.Object)
!flags.is(Flags.Param) && !flags.is(Flags.ParamAccessor) && !flags.is(Flags.FieldAccessor) && !isUndecompilableCaseClassMethod && !isInnerModuleObject
}
val stats1 = stats.collect {
case stat: Definition if keepDefinition(stat) => stat
case stat @ Import(_, _) => stat
case stat: Term => stat
}
def printBody(printSelf: Boolean) = {
this += " {"
indented {
if (printSelf) {
val Some(ValDef(name, tpt, _)) = self
indented {
val name1 = if (name == "_") "this" else name
this += " " += highlightValDef(name1) += ": "
printTypeTree(tpt)(using Some(cdef.symbol))
this += " =>"
}
}
this += lineBreak()
printTrees(stats1, lineBreak())
}
this += lineBreak() += "}"
}
self match {
case Some(ValDef(_, Singleton(_), _)) =>
if (stats1.nonEmpty)
printBody(printSelf = false)
case Some(ValDef(_, _, _)) =>
printBody(printSelf = true)
case _ =>
if (stats1.nonEmpty)
printBody(printSelf = false)
}
this
case tdef @ TypeDef(name, rhs) =>
printDefAnnotations(tdef)
this += highlightKeyword("type ")
printTargDef((tdef, tdef), isMember = true)
case vdef @ ValDef(name, tpt, rhs) =>
printDefAnnotations(vdef)
val flags = vdef.symbol.flags
if (flags.is(Flags.Implicit)) this += highlightKeyword("implicit ")
if (flags.is(Flags.Override)) this += highlightKeyword("override ")
if (flags.is(Flags.Final) && !flags.is(Flags.Object)) this += highlightKeyword("final ")
printProtectedOrPrivate(vdef)
if (flags.is(Flags.Lazy)) this += highlightKeyword("lazy ")
if (vdef.symbol.flags.is(Flags.Mutable)) this += highlightKeyword("var ")
else this += highlightKeyword("val ")
val name1 = splicedName(vdef.symbol).getOrElse(name)
this += highlightValDef(name1) += ": "
printTypeTree(tpt)
rhs match {
case Some(tree) =>
this += " = "
printTree(tree)
case None =>
this
}
case While(cond, body) =>
(cond, body) match {
case (Block(Block(Nil, body1) :: Nil, Block(Nil, cond1)), Literal(Constant.Unit())) =>
this += highlightKeyword("do ")
printTree(body1) += highlightKeyword(" while ")
inParens(printTree(cond1))
case _ =>
this += highlightKeyword("while ")
inParens(printTree(cond)) += " "
printTree(body)
}
case ddef @ DefDef(name, targs, argss, tpt, rhs) =>
printDefAnnotations(ddef)
val isConstructor = name == ""
val flags = ddef.symbol.flags
if (flags.is(Flags.Implicit)) this += highlightKeyword("implicit ")
if (flags.is(Flags.Inline)) this += highlightKeyword("inline ")
if (flags.is(Flags.Override)) this += highlightKeyword("override ")
if (flags.is(Flags.Final) && !flags.is(Flags.Object)) this += highlightKeyword("final ")
printProtectedOrPrivate(ddef)
val name1: String = if (isConstructor) "this" else splicedName(ddef.symbol).getOrElse(name)
this += highlightKeyword("def ") += highlightValDef(name1)
printTargsDefs(targs.zip(targs))
val it = argss.iterator
while (it.hasNext)
printArgsDefs(it.next())
if (!isConstructor) {
this += ": "
printTypeTree(tpt)
}
rhs match {
case Some(tree) =>
this += " = "
printTree(tree)
case None =>
}
this
case Ident("_") =>
this += "_"
case tree: Ident =>
splicedName(tree.symbol) match {
case Some(name) => this += highlightTypeDef(name)
case _ => printType(tree.tpe)
}
case Select(qual, name) =>
printQualTree(qual)
if (name != "" && name != "package")
this += "." += name
this
case Literal(const) =>
printConstant(const)
case This(id) =>
id match {
case Some(name) =>
this += name.stripSuffix("$") += "."
case None =>
}
this += "this"
case tree: New =>
this += "new "
printType(tree.tpe)
case NamedArg(name, arg) =>
this += name += " = "
printTree(arg)
case SpecialOp("throw", expr :: Nil) =>
this += "throw "
printTree(expr)
case Apply(fn, args) if fn.symbol == Symbol.requiredMethod("scala.internal.Quoted.exprQuote") =>
args.head match {
case Block(stats, expr) =>
this += "'{"
indented {
this += lineBreak()
printFlatBlock(stats, expr)
}
this += lineBreak() += "}"
case _ =>
this += "'{"
printTree(args.head)
this += "}"
}
case TypeApply(fn, args) if fn.symbol == Symbol.requiredMethod("scala.internal.Quoted.typeQuote") =>
this += "'["
printTypeTree(args.head)
this += "]"
case Apply(fn, arg :: Nil) if fn.symbol == Symbol.requiredMethod("scala.internal.Quoted.exprSplice") =>
this += "${"
printTree(arg)
this += "}"
case Apply(fn, args) =>
var argsPrefix = ""
fn match {
case Select(This(_), "") => this += "this" // call to constructor inside a constructor
case Select(qual, "apply") =>
if qual.tpe.isContextFunctionType then
argsPrefix += "using "
if qual.tpe.isErasedFunctionType then
argsPrefix += "erased "
printQualTree(fn)
case _ => printQualTree(fn)
}
val args1 = args match {
case init :+ Typed(Repeated(Nil, _), _) => init // drop empty var args at the end
case _ => args
}
inParens {
this += argsPrefix
printTrees(args1, ", ")
}
case TypeApply(fn, args) =>
printQualTree(fn)
fn match {
case Select(New(Applied(_, _)), "") =>
// type bounds already printed in `fn`
this
case _ =>
inSquare(printTrees(args, ", "))
}
case Super(qual, idOpt) =>
qual match {
case This(Some(name)) => this += name += "."
case This(None) =>
}
this += "super"
for (id <- idOpt)
inSquare(this += id)
this
case Typed(term, tpt) =>
tpt.tpe match {
case Types.Repeated(_) =>
printTree(term)
term match {
case Repeated(_, _) | Inlined(None, Nil, Repeated(_, _)) => this
case _ => this += ": " += highlightTypeDef("_*")
}
case _ =>
inParens {
printTree(term)
this += (if (scala.internal.Chars.isOperatorPart(sb.last)) " : " else ": ")
def printTypeOrAnnots(tpe: Type): Unit = tpe match {
case AnnotatedType(tp, annot) if tp == term.tpe =>
printAnnotation(annot)
case AnnotatedType(tp, annot) =>
printTypeOrAnnots(tp)
this += " "
printAnnotation(annot)
case tpe =>
printType(tpe)
}
printTypeOrAnnots(tpt.tpe)
}
}
case Assign(lhs, rhs) =>
printTree(lhs)
this += " = "
printTree(rhs)
case tree @ Lambda(params, body) => // must come before `Block`
inParens {
printArgsDefs(params)
this += (if tree.tpe.isContextFunctionType then " ?=> " else " => ")
printTree(body)
}
case Block(stats0, expr) =>
val stats = stats0.filter {
case tree: ValDef => !tree.symbol.flags.is(Flags.Object)
case _ => true
}
printFlatBlock(stats, expr)
case Inlined(_, bindings, expansion) =>
printFlatBlock(bindings, expansion)
case If(cond, thenp, elsep) =>
this += highlightKeyword("if ")
inParens(printTree(cond))
this += " "
printTree(thenp)
this+= highlightKeyword(" else ")
printTree(elsep)
case Match(selector, cases) =>
printQualTree(selector)
this += highlightKeyword(" match")
inBlock(printCases(cases, lineBreak()))
case GivenMatch(cases) =>
this += highlightKeyword("given match") // TODO: drop
inBlock(printCases(cases, lineBreak()))
case Try(body, cases, finallyOpt) =>
this += highlightKeyword("try ")
printTree(body)
if (cases.nonEmpty) {
this += highlightKeyword(" catch")
inBlock(printCases(cases, lineBreak()))
}
finallyOpt match {
case Some(t) =>
this += highlightKeyword(" finally ")
printTree(t)
case None =>
this
}
case Return(expr) =>
this += "return "
printTree(expr)
case Repeated(elems, _) =>
printTrees(elems, ", ")
case TypeBoundsTree(lo, hi) =>
this += "_ >: "
printTypeTree(lo)
this += " <: "
printTypeTree(hi)
case tpt: WildcardTypeTree =>
printType(tpt.tpe)
case tpt: TypeTree =>
printTypeTree(tpt)
case Closure(meth, _) =>
printTree(meth)
case _ =>
throw new MatchError(tree.showExtractors)
}
def printQualTree(tree: Tree): Buffer = tree match {
case _: If | _: Match | _: While | _: Try | _: Return =>
this += "("
printTree(tree)
this += ")"
case _ => printTree(tree)
}
def flatBlock(stats: List[Statement], expr: Term): (List[Statement], Term) = {
val flatStats = List.newBuilder[Statement]
def extractFlatStats(stat: Statement): Unit = stat match {
case Lambda(_, _) => // must come before `Block`
flatStats += stat
case Block(stats1, expr1) =>
val it = stats1.iterator
while (it.hasNext)
extractFlatStats(it.next())
extractFlatStats(expr1)
case Inlined(_, bindings, expansion) =>
val it = bindings.iterator
while (it.hasNext)
extractFlatStats(it.next())
extractFlatStats(expansion)
case Literal(Constant.Unit()) => // ignore
case stat => flatStats += stat
}
def extractFlatExpr(term: Term): Term = term match {
case Lambda(_, _) => // must come before `Block`
term
case Block(stats1, expr1) =>
val it = stats1.iterator
while (it.hasNext)
extractFlatStats(it.next())
extractFlatExpr(expr1)
case Inlined(_, bindings, expansion) =>
val it = bindings.iterator
while (it.hasNext)
extractFlatStats(it.next())
extractFlatExpr(expansion)
case term => term
}
val it = stats.iterator
while (it.hasNext)
extractFlatStats(it.next())
val flatExpr = extractFlatExpr(expr)
(flatStats.result(), flatExpr)
}
def printFlatBlock(stats: List[Statement], expr: Term)(using elideThis: Option[Symbol]): Buffer = {
val (stats1, expr1) = flatBlock(stats, expr)
val stats2 = stats1.filter {
case tree: TypeDef => !tree.symbol.annots.exists(_.symbol.maybeOwner == Symbol.requiredClass("scala.internal.Quoted.quoteTypeTag"))
case _ => true
}
if (stats2.isEmpty) {
printTree(expr1)
} else {
this += "{"
indented {
printStats(stats2, expr1)
}
this += lineBreak() += "}"
}
}
def printStats(stats: List[Tree], expr: Tree)(using eliseThis: Option[Symbol]): Unit = {
def printSeparator(next: Tree): Unit = {
// Avoid accidental application of opening `{` on next line with a double break
def rec(next: Tree): Unit = next match {
case Lambda(_, _) => this += lineBreak()
case Block(stats, _) if stats.nonEmpty => this += doubleLineBreak()
case Inlined(_, bindings, _) if bindings.nonEmpty => this += doubleLineBreak()
case Select(qual, _) => rec(qual)
case Apply(fn, _) => rec(fn)
case TypeApply(fn, _) => rec(fn)
case Typed(_, _) => this += doubleLineBreak()
case _ => this += lineBreak()
}
next match {
case term: Term =>
flatBlock(Nil, term) match {
case (next :: _, _) => rec(next)
case (Nil, next) => rec(next)
}
case _ => this += lineBreak()
}
}
def printSeparated(list: List[Tree]): Unit = list match {
case Nil =>
printTree(expr)
case x :: xs =>
printTree(x)
printSeparator(if (xs.isEmpty) expr else xs.head)
printSeparated(xs)
}
this += lineBreak()
printSeparated(stats)
}
def printList[T](xs: List[T], sep: String, print: T => Buffer): Buffer = {
def printSeparated(list: List[T]): Unit = list match {
case Nil =>
case x :: Nil => print(x)
case x :: xs =>
print(x)
this += sep
printSeparated(xs)
}
printSeparated(xs)
this
}
def printTrees(trees: List[Tree], sep: String)(using elideThis: Option[Symbol]): Buffer =
printList(trees, sep, (t: Tree) => printTree(t))
def printTypeTrees(trees: List[TypeTree], sep: String)(using elideThis: Option[Symbol] = None): Buffer =
printList(trees, sep, (t: TypeTree) => printTypeTree(t))
def printTypes(trees: List[Type], sep: String)(using elideThis: Option[Symbol]): Buffer = {
def printSeparated(list: List[Type]): Unit = list match {
case Nil =>
case x :: Nil => printType(x)
case x :: xs =>
printType(x)
this += sep
printSeparated(xs)
}
printSeparated(trees)
this
}
def printImportSelectors(selectors: List[ImportSelector]): Buffer = {
def printSeparated(list: List[ImportSelector]): Unit = list match {
case Nil =>
case x :: Nil => printImportSelector(x)
case x :: xs =>
printImportSelector(x)
this += ", "
printSeparated(xs)
}
this += "{"
printSeparated(selectors)
this += "}"
}
def printCases(cases: List[CaseDef], sep: String): Buffer = {
def printSeparated(list: List[CaseDef]): Unit = list match {
case Nil =>
case x :: Nil => printCaseDef(x)
case x :: xs =>
printCaseDef(x)
this += sep
printSeparated(xs)
}
printSeparated(cases)
this
}
def printTypeCases(cases: List[TypeCaseDef], sep: String): Buffer = {
def printSeparated(list: List[TypeCaseDef]): Unit = list match {
case Nil =>
case x :: Nil => printTypeCaseDef(x)
case x :: xs =>
printTypeCaseDef(x)
this += sep
printSeparated(xs)
}
printSeparated(cases)
this
}
def printPatterns(cases: List[Tree], sep: String): Buffer = {
def printSeparated(list: List[Tree]): Unit = list match {
case Nil =>
case x :: Nil => printPattern(x)
case x :: xs =>
printPattern(x)
this += sep
printSeparated(xs)
}
printSeparated(cases)
this
}
def printTypesOrBounds(types: List[Type], sep: String)(using elideThis: Option[Symbol]): Buffer = {
def printSeparated(list: List[Type]): Unit = list match {
case Nil =>
case x :: Nil => printType(x)
case x :: xs =>
printType(x)
this += sep
printSeparated(xs)
}
printSeparated(types)
this
}
def printTargsDefs(targs: List[(TypeDef, TypeDef)], isDef:Boolean = true)(using elideThis: Option[Symbol]): Unit = {
if (!targs.isEmpty) {
def printSeparated(list: List[(TypeDef, TypeDef)]): Unit = list match {
case Nil =>
case x :: Nil => printTargDef(x, isDef = isDef)
case x :: xs =>
printTargDef(x, isDef = isDef)
this += ", "
printSeparated(xs)
}
inSquare(printSeparated(targs))
}
}
def printTargDef(arg: (TypeDef, TypeDef), isMember: Boolean = false, isDef:Boolean = true)(using elideThis: Option[Symbol]): Buffer = {
val (argDef, argCons) = arg
if (isDef) {
if (argDef.symbol.flags.is(Flags.Covariant)) {
this += highlightValDef("+")
} else if (argDef.symbol.flags.is(Flags.Contravariant)) {
this += highlightValDef("-")
}
}
this += argCons.name
argCons.rhs match {
case rhs: TypeBoundsTree => printBoundsTree(rhs)
case rhs: WildcardTypeTree =>
printType(rhs.tpe)
case rhs @ LambdaTypeTree(tparams, body) =>
def printParam(t: Tree /*TypeTree | TypeBoundsTree*/): Unit = t match {
case t: TypeBoundsTree => printBoundsTree(t)
case t: TypeTree => printTypeTree(t)
}
def printSeparated(list: List[TypeDef]): Unit = list match {
case Nil =>
case x :: Nil =>
this += x.name
printParam(x.rhs)
case x :: xs =>
this += x.name
printParam(x.rhs)
this += ", "
printSeparated(xs)
}
inSquare(printSeparated(tparams))
if (isMember) {
body match {
case MatchTypeTree(Some(bound), _, _) =>
this += " <: "
printTypeTree(bound)
case _ =>
}
this += " = "
printTypeOrBoundsTree(body)
}
else this
case rhs: TypeTree =>
this += " = "
printTypeTree(rhs)
}
}
def printArgsDefs(args: List[ValDef])(using elideThis: Option[Symbol]): Unit = {
val argFlags = args match {
case Nil => Flags.EmptyFlags
case arg :: _ => arg.symbol.flags
}
if (argFlags.is(Flags.Erased | Flags.Given)) {
if (argFlags.is(Flags.Given)) this += " given"
if (argFlags.is(Flags.Erased)) this += " erased"
this += " "
}
inParens {
if (argFlags.is(Flags.Implicit) && !argFlags.is(Flags.Given)) this += "implicit "
def printSeparated(list: List[ValDef]): Unit = list match {
case Nil =>
case x :: Nil => printParamDef(x)
case x :: xs =>
printParamDef(x)
this += ", "
printSeparated(xs)
}
printSeparated(args)
}
}
def printAnnotations(trees: List[Term])(using elideThis: Option[Symbol]): Buffer = {
def printSeparated(list: List[Term]): Unit = list match {
case Nil =>
case x :: Nil => printAnnotation(x)
case x :: xs =>
printAnnotation(x)
this += " "
printSeparated(xs)
}
printSeparated(trees)
this
}
def printParamDef(arg: ValDef)(using elideThis: Option[Symbol]): Unit = {
val name = splicedName(arg.symbol).getOrElse(arg.symbol.name)
val sym = arg.symbol.owner
if sym.isDefDef && sym.name == "" then
val ClassDef(_, _, _, _, _, body) = sym.owner.tree
body.collectFirst {
case vdef @ ValDef(`name`, _, _) if vdef.symbol.flags.is(Flags.ParamAccessor) =>
if (!vdef.symbol.flags.is(Flags.Local)) {
var printedPrefix = false
if (vdef.symbol.flags.is(Flags.Override)) {
this += "override "
printedPrefix = true
}
printedPrefix |= printProtectedOrPrivate(vdef)
if (vdef.symbol.flags.is(Flags.Mutable)) this += highlightValDef("var ")
else if (printedPrefix || !vdef.symbol.flags.is(Flags.CaseAccessor)) this += highlightValDef("val ")
}
}
end if
this += highlightValDef(name) += ": "
printTypeTree(arg.tpt)
}
def printCaseDef(caseDef: CaseDef): Buffer = {
this += highlightValDef("case ")
printPattern(caseDef.pattern)
caseDef.guard match {
case Some(t) =>
this += " if "
printTree(t)
case None =>
}
this += highlightValDef(" =>")
indented {
caseDef.rhs match {
case Block(stats, expr) =>
printStats(stats, expr)(using None)
case body =>
this += lineBreak()
printTree(body)
}
}
this
}
def printTypeCaseDef(caseDef: TypeCaseDef): Buffer = {
this += highlightValDef("case ")
printTypeTree(caseDef.pattern)
this += highlightValDef(" => ")
printTypeTree(caseDef.rhs)
this
}
def printPattern(pattern: Tree): Buffer = pattern match {
case Ident("_") =>
this += "_"
case Bind(name, Ident("_")) =>
this += name
case Bind(name, Typed(Ident("_"), tpt)) =>
this += highlightValDef(name) += ": "
printTypeTree(tpt)
case Bind(name, pattern) =>
this += name += " @ "
printPattern(pattern)
case Unapply(fun, implicits, patterns) =>
val fun2 = fun match {
case TypeApply(fun2, _) => fun2
case _ => fun
}
fun2 match {
case Select(extractor, "unapply" | "unapplySeq") =>
printTree(extractor)
case Ident("unapply" | "unapplySeq") =>
this += fun.symbol.owner.fullName.stripSuffix("$")
case _ =>
throw new MatchError(fun.showExtractors)
}
inParens(printPatterns(patterns, ", "))
case Alternatives(trees) =>
inParens(printPatterns(trees, " | "))
case Typed(Ident("_"), tpt) =>
this += "_: "
printTypeTree(tpt)
case v: Term =>
printTree(v)
case _ =>
throw new MatchError(pattern.showExtractors)
}
inline private val qc = '\''
inline private val qSc = '"'
def printConstant(const: Constant): Buffer = const match {
case Constant.Unit() => this += highlightLiteral("()")
case Constant.Null() => this += highlightLiteral("null")
case Constant.Boolean(v) => this += highlightLiteral(v.toString)
case Constant.Byte(v) => this += highlightLiteral(v.toString)
case Constant.Short(v) => this += highlightLiteral(v.toString)
case Constant.Int(v) => this += highlightLiteral(v.toString)
case Constant.Long(v) => this += highlightLiteral(v.toString + "L")
case Constant.Float(v) => this += highlightLiteral(v.toString + "f")
case Constant.Double(v) => this += highlightLiteral(v.toString)
case Constant.Char(v) => this += highlightString(s"${qc}${escapedChar(v)}${qc}")
case Constant.String(v) => this += highlightString(s"${qSc}${escapedString(v)}${qSc}")
case Constant.ClassOf(v) =>
this += "classOf"
inSquare(printType(v))
}
def printTypeOrBoundsTree(tpt: Tree)(using elideThis: Option[Symbol] = None): Buffer = tpt match {
case TypeBoundsTree(lo, hi) =>
this += "_ >: "
printTypeTree(lo)
this += " <: "
printTypeTree(hi)
case tpt: WildcardTypeTree =>
printType(tpt.tpe)
case tpt: TypeTree =>
printTypeTree(tpt)
}
/** Print type tree
*
* @param elideThis Shoud printing elide `C.this` for the given class `C`?
* None means no eliding.
*
* Self type annotation and types in parent list should elide current class
* prefix `C.this` to avoid type checking errors.
*/
def printTypeTree(tree: TypeTree)(using elideThis: Option[Symbol] = None): Buffer = tree match {
case Inferred() =>
// TODO try to move this logic into `printType`
def printTypeAndAnnots(tpe: Type): Buffer = tpe match {
case AnnotatedType(tp, annot) =>
printTypeAndAnnots(tp)
this += " "
printAnnotation(annot)
case tpe: TypeRef if tpe.typeSymbol == Symbol.requiredClass("scala.runtime.Null$") || tpe.typeSymbol == Symbol.requiredClass("scala.runtime.Nothing$") =>
// scala.runtime.Null$ and scala.runtime.Nothing$ are not modules, those are their actual names
printType(tpe)
case tpe: TermRef if tpe.termSymbol.isClassDef && tpe.termSymbol.name.endsWith("$") =>
printType(tpe)
this += ".type"
case tpe: TypeRef if tpe.typeSymbol.isClassDef && tpe.typeSymbol.name.endsWith("$") =>
printType(tpe)
this += ".type"
case tpe @ TermRef(sym, _) =>
printType(tpe)
this += ".type"
case tpe => printType(tpe)
}
printTypeAndAnnots(tree.tpe)
case TypeIdent(name) =>
printType(tree.tpe)
case TypeSelect(qual, name) =>
printTree(qual) += "." += highlightTypeDef(name)
case Projection(qual, name) =>
printTypeTree(qual) += "#" += highlightTypeDef(name)
case Singleton(ref) =>
printTree(ref)
ref match {
case Literal(_) => this
case _ => this += ".type"
}
case Refined(tpt, refinements) =>
printTypeTree(tpt)
inBlock(printTrees(refinements, "; "))
case Applied(tpt, args) =>
printTypeTree(tpt)
inSquare(printTrees(args, ", "))
case Annotated(tpt, annot) =>
val Annotation(ref, args) = annot
ref.tpe match {
case tpe: TypeRef if tpe.typeSymbol == Symbol.requiredClass("scala.annotation.internal.Repeated") =>
val Types.Sequence(tp) = tpt.tpe
printType(tp)
this += highlightTypeDef("*")
case _ =>
printTypeTree(tpt)
this += " "
printAnnotation(annot)
}
case MatchTypeTree(bound, selector, cases) =>
printTypeTree(selector)
this += highlightKeyword(" match ")
inBlock(printTypeCases(cases, lineBreak()))
case ByName(result) =>
this += highlightTypeDef("=> ")
printTypeTree(result)
case LambdaTypeTree(tparams, body) =>
printTargsDefs(tparams.zip(tparams), isDef = false)
this += highlightTypeDef(" => ")
printTypeOrBoundsTree(body)
case TypeBind(name, _) =>
this += highlightTypeDef(name)
case TypeBlock(_, tpt) =>
printTypeTree(tpt)
case _ =>
throw new MatchError(tree.showExtractors)
}
/** Print type
*
* @param elideThis Shoud printing elide `C.this` for the given class `C`?
* None means no eliding.
*
* Self type annotation and types in parent list should elide current class
* prefix `C.this` to avoid type checking errors.
*/
def printType(tpe: Type)(using elideThis: Option[Symbol] = None): Buffer = tpe match {
case ConstantType(const) =>
printConstant(const)
case tpe: TypeRef =>
val sym = tpe.typeSymbol
tpe.qualifier match {
case ThisType(tp) if tp.typeSymbol == defn.RootClass || tp.typeSymbol == defn.EmptyPackageClass =>
case NoPrefix() =>
if (sym.owner.flags.is(Flags.Package)) {
// TODO should these be in the prefix? These are at least `scala`, `java` and `scala.collection`.
val packagePath = sym.owner.fullName.stripPrefix("").stripPrefix("").stripPrefix(".")
if (packagePath != "")
this += packagePath += "."
}
case prefix: TermRef if prefix.termSymbol.isClassDef =>
printType(prefix)
this += "#"
case prefix: TypeRef if prefix.typeSymbol.isClassDef =>
printType(prefix)
this += "#"
case ThisType(TermRef(cdef, _)) if elideThis.nonEmpty && cdef == elideThis.get =>
case ThisType(TypeRef(cdef, _)) if elideThis.nonEmpty && cdef == elideThis.get =>
case prefix: Type =>
printType(prefix)
this += "."
}
this += highlightTypeDef(sym.name.stripSuffix("$"))
case TermRef(prefix, name) =>
prefix match {
case NoPrefix() =>
this += highlightTypeDef(name)
case ThisType(tp) if tp.typeSymbol == defn.RootClass || tp.typeSymbol == defn.EmptyPackageClass =>
this += highlightTypeDef(name)
case _ =>
printType(prefix)
if (name != "package")
this += "." += highlightTypeDef(name)
this
}
case tpe @ Refinement(_, _, _) =>
printRefinement(tpe)
case AppliedType(tp, args) =>
tp match {
case tp: TypeLambda =>
this += "("
printType(tp)
this += ")"
inSquare(printTypesOrBounds(args, ", "))
case tp: TypeRef if tp.typeSymbol == Symbol.requiredClass("scala.") =>
this += "_*"
case _ =>
printType(tp)
inSquare(printTypesOrBounds(args, ", "))
}
case AnnotatedType(tp, annot) =>
val Annotation(ref, args) = annot
printType(tp)
this += " "
printAnnotation(annot)
case AndType(left, right) =>
printType(left)
this += highlightTypeDef(" & ")
printType(right)
case OrType(left, right) =>
printType(left)
this += highlightTypeDef(" | ")
printType(right)
case MatchType(bound, scrutinee, cases) =>
printType(scrutinee)
this += highlightKeyword(" match ")
inBlock(printTypes(cases, lineBreak()))
case ByNameType(tp) =>
this += highlightTypeDef(" => ")
printType(tp)
case ThisType(tp) =>
tp match {
case tp: TypeRef if !tp.typeSymbol.flags.is(Flags.Object) =>
printFullClassName(tp)
this += highlightTypeDef(".this")
case TypeRef(prefix, name) if name.endsWith("$") =>
prefix match {
case NoPrefix() =>
case ThisType(tp) if tp.typeSymbol == defn.RootClass || tp.typeSymbol == defn.EmptyPackageClass =>
case _ =>
printType(prefix)
this += "."
}
this += highlightTypeDef(name.stripSuffix("$"))
case _ =>
printType(tp)
}
case SuperType(thistpe, supertpe) =>
printType(supertpe)
this += highlightTypeDef(".super")
case TypeLambda(paramNames, tparams, body) =>
inSquare(printMethodicTypeParams(paramNames, tparams))
this += highlightTypeDef(" => ")
printType(body)
case ParamRef(lambda, idx) =>
lambda match {
case MethodType(params, _, _) => this += params(idx)
case PolyType(params, _, _) => this += params(idx)
case TypeLambda(params, _, _) => this += params(idx)
}
case RecursiveType(tpe) =>
printType(tpe)
case RecursiveThis(_) =>
this += highlightTypeDef("this")
case tpe: MethodType =>
this += "("
printList(tpe.paramNames.zip(tpe.paramTypes), ", ",
(x: (String, Type)) => (this += x._1 += ": ").printType(x._2))
this += ")"
printType(tpe.resType)
case tpe: PolyType =>
this += "["
printList(tpe.paramNames.zip(tpe.paramBounds), ", ",
(x: (String, TypeBounds)) => (this += x._1 += " ").printType(x._2))
this += "]"
printType(tpe.resType)
case tpe: TypeLambda =>
this += "["
printList(tpe.paramNames.zip(tpe.paramBounds), ", ",
(x: (String, TypeBounds)) => (this += x._1 += " ").printType(x._2))
this += "] => "
printType(tpe.resType)
case tpe@TypeBounds(lo, hi) =>
this += "_ >: "
printType(lo)
this += " <: "
printType(hi)
case _ =>
throw new MatchError(tpe.showExtractors)
}
def printImportSelector(sel: ImportSelector): Buffer = sel match {
case SimpleSelector(name) => this += name
case OmitSelector(name) => this += name += " => _"
case RenameSelector(name, newName) => this += name += " => " += newName
}
def printDefinitionName(tree: Definition): Buffer = tree match {
case ValDef(name, _, _) => this += highlightValDef(name)
case DefDef(name, _, _, _, _) => this += highlightValDef(name)
case ClassDef(name, _, _, _, _, _) => this += highlightTypeDef(name.stripSuffix("$"))
case TypeDef(name, _) => this += highlightTypeDef(name)
}
def printAnnotation(annot: Term)(using elideThis: Option[Symbol]): Buffer = {
val Annotation(ref, args) = annot
if (annot.symbol.maybeOwner == Symbol.requiredClass("scala.internal.quoted.showName")) this
else {
this += "@"
printTypeTree(ref)
if (args.isEmpty)
this
else
inParens(printTrees(args, ", "))
}
}
def printDefAnnotations(definition: Definition)(using elideThis: Option[Symbol]): Buffer = {
val annots = definition.symbol.annots.filter {
case Annotation(annot, _) =>
val sym = annot.tpe.typeSymbol
sym != Symbol.requiredClass("scala.forceInline") &&
sym.maybeOwner != Symbol.requiredPackage("scala.annotation.internal")
case x => throw new MatchError(x.showExtractors)
}
printAnnotations(annots)
if (annots.nonEmpty) this += " "
else this
}
def printRefinement(tpe: Type)(using elideThis: Option[Symbol]): Buffer = {
def printMethodicType(tp: Type): Unit = tp match {
case tp @ MethodType(paramNames, params, res) =>
inParens(printMethodicTypeParams(paramNames, params))
printMethodicType(res)
case tp @ TypeLambda(paramNames, params, res) =>
inSquare(printMethodicTypeParams(paramNames, params))
printMethodicType(res)
case ByNameType(t) =>
this += ": "
printType(t)
case tp: Type =>
this += ": "
printType(tp)
}
def rec(tp: Type): Unit = tp match {
case Refinement(parent, name, info) =>
rec(parent)
indented {
this += lineBreak()
info match {
case info: TypeBounds =>
this += highlightKeyword("type ") += highlightTypeDef(name)
printBounds(info)
case ByNameType(_) | MethodType(_, _, _) | TypeLambda(_, _, _) =>
this += highlightKeyword("def ") += highlightTypeDef(name)
printMethodicType(info)
case info: Type =>
this += highlightKeyword("val ") += highlightValDef(name)
printMethodicType(info)
}
}
case tp =>
printType(tp)
this += " {"
}
rec(tpe)
this += lineBreak() += "}"
}
def printMethodicTypeParams(paramNames: List[String], params: List[Type])(using elideThis: Option[Symbol]): Unit = {
def printInfo(info: Type) = info match {
case info: TypeBounds => printBounds(info)
case info: Type =>
this += ": "
printType(info)
}
def printSeparated(list: List[(String, Type)]): Unit = list match {
case Nil =>
case (name, info) :: Nil =>
this += name
printInfo(info)
case (name, info) :: xs =>
this += name
printInfo(info)
this += ", "
printSeparated(xs)
}
printSeparated(paramNames.zip(params))
}
def printBoundsTree(bounds: TypeBoundsTree)(using elideThis: Option[Symbol]): Buffer = {
bounds.low match {
case Inferred() =>
case low =>
this += " >: "
printTypeTree(low)
}
bounds.hi match {
case Inferred() => this
case hi =>
this += " <: "
printTypeTree(hi)
}
}
def printBounds(bounds: TypeBounds)(using elideThis: Option[Symbol]): Buffer = {
this += " >: "
printType(bounds.low)
this += " <: "
printType(bounds.hi)
}
def printProtectedOrPrivate(definition: Definition): Boolean = {
var prefixWasPrinted = false
def printWithin(within: Type) = within match {
case TypeRef(_, name) => this += name
case _ => printFullClassName(within)
}
if (definition.symbol.flags.is(Flags.Protected)) {
this += highlightKeyword("protected")
definition.symbol.protectedWithin match {
case Some(within) =>
inSquare(printWithin(within))
case _ =>
}
prefixWasPrinted = true
} else {
definition.symbol.privateWithin match {
case Some(within) =>
this += highlightKeyword("private")
inSquare(printWithin(within))
prefixWasPrinted = true
case _ =>
}
}
if (prefixWasPrinted)
this += " "
prefixWasPrinted
}
def printFullClassName(tp: Type): Unit = {
def printClassPrefix(prefix: Type): Unit = prefix match {
case TypeRef(prefix2, name) =>
printClassPrefix(prefix2)
this += name += "."
case _ =>
}
val TypeRef(prefix, name) = tp
printClassPrefix(prefix)
this += name
}
def +=(x: Boolean): this.type = { sb.append(x); this }
def +=(x: Byte): this.type = { sb.append(x); this }
def +=(x: Short): this.type = { sb.append(x); this }
def +=(x: Int): this.type = { sb.append(x); this }
def +=(x: Long): this.type = { sb.append(x); this }
def +=(x: Float): this.type = { sb.append(x); this }
def +=(x: Double): this.type = { sb.append(x); this }
def +=(x: Char): this.type = { sb.append(x); this }
def +=(x: String): this.type = { sb.append(x); this }
private def escapedChar(ch: Char): String = (ch: @switch) match {
case '\b' => "\\b"
case '\t' => "\\t"
case '\n' => "\\n"
case '\f' => "\\f"
case '\r' => "\\r"
case '"' => "\\\""
case '\'' => "\\\'"
case '\\' => "\\\\"
case _ => if (ch.isControl) "\\0" + Integer.toOctalString(ch) else String.valueOf(ch)
}
private def escapedString(str: String): String = str flatMap escapedChar
}
private[this] val names = collection.mutable.Map.empty[Symbol, String]
private[this] val namesIndex = collection.mutable.Map.empty[String, Int]
private def splicedName(sym: Symbol): Option[String] = {
sym.annots.find(_.symbol.owner == Symbol.requiredClass("scala.internal.quoted.showName")).flatMap {
case Apply(_, Literal(Constant.String(c)) :: Nil) => Some(c)
case Apply(_, Inlined(_, _, Literal(Constant.String(c))) :: Nil) => Some(c)
case annot => None
}.orElse {
if sym.owner.isClassDef then None
else names.get(sym).orElse {
val name0 = sym.name
val index = namesIndex.getOrElse(name0, 1)
namesIndex(name0) = index + 1
val name =
if index == 1 then name0
else s"`$name0${index.toString.toCharArray.map {x => (x - '0' + '₀').toChar}.mkString}`"
names(sym) = name
Some(name)
}
}
}
private object SpecialOp {
def unapply(arg: Tree): Option[(String, List[Term])] = arg match {
case arg @ Apply(fn, args) =>
fn.tpe match {
case tpe @ TermRef(ThisType(TypeRef(_, name)), name2) if name == "" =>
Some((name2, args))
case _ => None
}
case _ => None
}
}
private object Annotation {
def unapply(arg: Tree): Option[(TypeTree, List[Term])] = arg match {
case New(annot) => Some((annot, Nil))
case Apply(Select(New(annot), ""), args) => Some((annot, args))
case Apply(TypeApply(Select(New(annot), ""), targs), args) => Some((annot, args))
case _ => None
}
}
// TODO Provide some of these in scala.tasty.Reflection.scala and implement them using checks on symbols for performance
private object Types {
object Sequence {
def unapply(tpe: Type): Option[Type] = tpe match {
case AppliedType(seq, (tp: Type) :: Nil)
if seq.typeSymbol == Symbol.requiredClass("scala.collection.Seq") || seq.typeSymbol == Symbol.requiredClass("scala.collection.immutable.Seq") =>
Some(tp)
case _ => None
}
}
object Repeated {
def unapply(tpe: Type): Option[Type] = tpe match {
case AppliedType(rep, (tp: Type) :: Nil) if rep.typeSymbol == Symbol.requiredClass("scala.") => Some(tp)
case _ => None
}
}
}
object PackageObject {
def unapply(tree: Tree): Option[Tree] = tree match {
case PackageClause(_, ValDef("package", _, _) :: body :: Nil) => Some(body)
case _ => None
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy