dotty.tools.dotc.semanticdb.PPrint.scala Maven / Gradle / Ivy
package dotty.tools.dotc.semanticdb
import dotty.tools.dotc.{semanticdb => s}
import scala.collection.mutable
import dotty.tools.dotc.semanticdb.Scala3.given
import SymbolInformation.Kind.*
import dotty.tools.dotc.util.SourceFile
class SymbolInformationPrinter (symtab: PrinterSymtab):
val notes = InfoNotes()
val infoPrinter = InfoPrinter(notes)
def pprintSymbolInformation(info: SymbolInformation): String =
val sb = new StringBuilder()
sb.append(info.symbol).append(" => ")
sb.append(infoPrinter.pprint(info))
sb.toString
class InfoNotes:
private val noteSymtab = mutable.Map[String, SymbolInformation]()
def enter(info: SymbolInformation) =
if (symtab.info(info.symbol).isEmpty && info.kind != UNKNOWN_KIND)
noteSymtab(info.symbol) = info
def visit(sym: String): SymbolInformation =
val symtabInfo = noteSymtab.get(sym).orElse(symtab.info(sym))
symtabInfo.getOrElse {
val displayName = if sym.isGlobal then sym.desc.value else sym
SymbolInformation(symbol = sym, displayName = displayName)
}
end InfoNotes
class InfoPrinter(notes: InfoNotes):
private enum SymbolStyle:
case Reference, Definition
def pprint(info: SymbolInformation): String =
val sb = new StringBuilder()
val annotStr = info.annotations.map(pprint).mkString(" ")
if annotStr.nonEmpty then
sb.append(annotStr + " ")
sb.append(accessString(info.access))
if info.isAbstract then sb.append("abstract ")
if info.isFinal then sb.append("final ")
if info.isSealed then sb.append("sealed ")
if info.isImplicit then sb.append("implicit ")
if info.isLazy then sb.append("lazy ")
if info.isCase then sb.append("case ")
if info.isCovariant then sb.append("covariant ")
if info.isContravariant then sb.append("contravariant ")
if info.isVal then sb.append("val ")
if info.isVar then sb.append("var ")
if info.isStatic then sb.append("static ")
if info.isPrimary then sb.append("primary ")
if info.isEnum then sb.append("enum ")
if info.isDefault then sb.append("default ")
if info.isGiven then sb.append("given ")
if info.isInline then sb.append("inline ")
if info.isOpen then sb.append("open ")
if info.isTransparent then sb.append("transparent ")
if info.isInfix then sb.append("infix ")
if info.isOpaque then sb.append("opaque ")
info.kind match
case LOCAL => sb.append("local ")
case FIELD => sb.append("field ")
case METHOD => sb.append("method ")
case CONSTRUCTOR => sb.append("ctor ")
case MACRO => sb.append("macro ")
case TYPE => sb.append("type ")
case PARAMETER => sb.append("param ")
case SELF_PARAMETER => sb.append("selfparam ")
case TYPE_PARAMETER => sb.append("typeparam ")
case OBJECT => sb.append("object ")
case PACKAGE => sb.append("package ")
case PACKAGE_OBJECT => sb.append("package object ")
case CLASS => sb.append("class ")
case TRAIT => sb.append("trait ")
case INTERFACE => sb.append("interface ")
case UNKNOWN_KIND | Unrecognized(_) => sb.append("unknown ")
sb.append(s"${info.displayName}${info.prefixBeforeTpe}${pprint(info.signature)}")
info.overriddenSymbols match
case Nil => ()
case all => sb.append(s" <: ${all.mkString(", ")}")
sb.toString
private def pprintDef(info: SymbolInformation) =
notes.enter(info)
pprint(info.symbol, SymbolStyle.Definition)
def pprintRef(sym: String): String = pprint(sym, SymbolStyle.Reference)
private def pprintDef(sym: String): String = pprint(sym, SymbolStyle.Definition)
private def pprint(sym: String, style: SymbolStyle): String =
val info = notes.visit(sym)
style match
case SymbolStyle.Reference =>
info.displayName
case SymbolStyle.Definition =>
pprint(info)
private def pprint(sig: Signature): String =
sig match
case ClassSignature(tparams, parents, self, decls) =>
val sb = new StringBuilder()
if (tparams.infos.nonEmpty)
sb.append(tparams.infos.map(pprintDef).mkString("[", ", ", "] "))
if (parents.nonEmpty)
sb.append(parents.map(pprint).mkString("extends ", " with ", " "))
if (self.isDefined || decls.infos.nonEmpty) {
val selfStr = if (self.isDefined) s"self: ${pprint(self)} =>" else ""
val declsStr = if (decls.infos.nonEmpty) s"+${decls.infos.length} decls" else ""
sb.append(s"{ ${selfStr} ${declsStr} }")
}
sb.toString
case MethodSignature(tparams, paramss, res) =>
val sb = new StringBuilder()
if (tparams.infos.nonEmpty)
sb.append(tparams.infos.map(pprintDef).mkString("[", ", ", "]"))
paramss.foreach { params =>
val paramsStr = params.infos.map(pprintDef).mkString("(", ", ", ")")
sb.append(paramsStr)
}
sb.append(s": ${pprint(res)}")
sb.toString
case TypeSignature(tparams, lo, hi) =>
val sb = new StringBuilder()
if (tparams.infos.nonEmpty)
sb.append(tparams.infos.map(pprintDef).mkString("[", ", ", "]"))
if (lo == hi) {
sb.append(s" = ${pprint(lo)}")
} else {
lo match
case TypeRef(Type.Empty, "scala/Nothing#", Nil) => ()
case lo => sb.append(s" >: ${pprint(lo)}")
hi match
case TypeRef(Type.Empty, "scala/Any#", Nil) => ()
case TypeRef(Type.Empty, "java/lang/Object#", Nil) => ()
case hi => sb.append(s" <: ${pprint(hi)}")
}
sb.toString
case ValueSignature(tpe) =>
pprint(tpe)
case _ =>
">"
protected def pprint(tpe: Type): String = {
def prefix(tpe: Type): String = tpe match
case TypeRef(pre, sym, args) =>
val preStr = pre match {
case _: SingleType | _: ThisType | _: SuperType =>
s"${prefix(pre)}."
case Type.Empty => ""
case _ =>
s"${prefix(pre)}#"
}
val argsStr = if (args.nonEmpty) args.map(normal).mkString("[", ", ", "]") else ""
s"${preStr}${pprintRef(sym)}${argsStr}"
case SingleType(pre, sym) =>
pre match {
case Type.Empty => pprintRef(sym)
case _ =>
s"${prefix(pre)}.${pprintRef(sym)}"
}
case ThisType(sym) =>
s"${pprintRef(sym)}.this"
case SuperType(pre, sym) =>
s"${prefix(pre)}.super[${pprintRef(sym)}]"
case ConstantType(const) =>
pprint(const)
case IntersectionType(types) =>
types.map(normal).mkString(" & ")
case UnionType(types) =>
types.map(normal).mkString(" | ")
case WithType(types) =>
types.map(normal).mkString(" with ")
case StructuralType(utpe, decls) =>
val declsStr =
if (decls.infos.nonEmpty)
s"{ ${decls.infos.map(pprintDef).mkString("; ")} }"
else "{}"
s"${normal(utpe)} ${declsStr}"
case AnnotatedType(anns, utpe) =>
s"${normal(utpe)} ${anns.map(pprint).mkString(" ")}"
case ExistentialType(utpe, decls) =>
val sdecls = decls.infos.map(pprintDef).mkString("; ")
val sutpe = normal(utpe)
s"${sutpe} forSome { ${sdecls} }"
case UniversalType(tparams, utpe) =>
val params = tparams.infos.map(_.displayName).mkString("[", ", ", "]")
val resType = normal(utpe)
s"${params} => ${resType}"
case ByNameType(utpe) =>
s"=> ${normal(utpe)}"
case RepeatedType(utpe) =>
s"${normal(utpe)}*"
case MatchType(scrutinee, cases) =>
val casesStr = cases.map { caseType =>
s"${pprint(caseType.key)} => ${pprint(caseType.body)}"
}.mkString(", ")
s"${pprint(scrutinee)} match { ${casesStr} }"
case LambdaType(tparams, res) =>
val params = tparams.infos.map(_.displayName).mkString("[", ", ", "]")
val resType = normal(res)
s"$params =>> $resType"
case x =>
">"
def normal(tpe: Type): String = tpe match
case _: SingleType | _: ThisType | _: SuperType =>
s"${prefix(tpe)}.type"
case _ =>
prefix(tpe)
normal(tpe)
}
private def pprint(ann: Annotation): String =
ann.tpe match {
case Type.Empty => s"@>"
case tpe => s"@${pprint(tpe)}"
}
protected def pprint(const: Constant): String = const match {
case Constant.Empty =>
">"
case UnitConstant() =>
"()"
case BooleanConstant(true) =>
"true"
case BooleanConstant(false) =>
"false"
case ByteConstant(value) =>
value.toByte.toString
case ShortConstant(value) =>
value.toShort.toString
case CharConstant(value) =>
s"'${value.toChar.toString}'"
case IntConstant(value) =>
value.toString
case LongConstant(value) =>
s"${value.toString}L"
case FloatConstant(value) =>
s"${value.toString}f"
case DoubleConstant(value) =>
value.toString
case StringConstant(value) =>
"\"" + value + "\""
case NullConstant() =>
"null"
}
private def accessString(access: Access): String =
access match
case Access.Empty => ""
case _: PublicAccess => ""
case _: PrivateAccess => "private "
case _: ProtectedAccess => "protected "
case _: PrivateThisAccess => "private[this] "
case _: ProtectedThisAccess => "protected[this] "
case PrivateWithinAccess(ssym) =>
s"private[${ssym}] "
case ProtectedWithinAccess(ssym) =>
s"protected[${ssym}] "
extension (scope: Scope)
private def infos: List[SymbolInformation] =
if (scope.symlinks.nonEmpty)
scope.symlinks.map(symbol => SymbolInformation(symbol = symbol)).toList
else
scope.hardlinks.toList
extension (scope: Option[Scope])
private def infos: List[SymbolInformation] = scope match {
case Some(s) => s.infos
case None => Nil
}
end InfoPrinter
end SymbolInformationPrinter
extension (info: SymbolInformation)
def prefixBeforeTpe: String = {
info.kind match {
case LOCAL | FIELD | PARAMETER | SELF_PARAMETER | UNKNOWN_KIND | Unrecognized(_) =>
": "
case METHOD | CONSTRUCTOR | MACRO | TYPE | TYPE_PARAMETER | OBJECT | PACKAGE |
PACKAGE_OBJECT | CLASS | TRAIT | INTERFACE =>
" "
}
}
trait PrinterSymtab:
def info(symbol: String): Option[SymbolInformation]
object PrinterSymtab:
def fromTextDocument(doc: TextDocument): PrinterSymtab =
val map = doc.symbols.map(info => (info.symbol, info)).toMap
new PrinterSymtab {
override def info(symbol: String): Option[SymbolInformation] = map.get(symbol)
}
def processRange(sb: StringBuilder, range: Range): Unit =
sb.append('[')
.append(range.startLine).append(':').append(range.startCharacter)
.append("..")
.append(range.endLine).append(':').append(range.endCharacter)
.append("):")
class SyntheticPrinter(symtab: PrinterSymtab, source: SourceFile) extends SymbolInformationPrinter(symtab):
def pprint(synth: Synthetic): String =
val sb = new StringBuilder()
val notes = InfoNotes()
val treePrinter = TreePrinter(source, synth.range, notes)
synth.range match
case Some(range) =>
processRange(sb, range)
sb.append(source.substring(range))
case None =>
sb.append("[):")
sb.append(" => ")
sb.append(treePrinter.pprint(synth.tree))
sb.toString
extension (source: SourceFile)
private def substring(range: Option[s.Range]): String =
range match
case Some(range) => source.substring(range)
case None => ""
private def substring(range: s.Range): String =
/** get the line length of a given line */
def lineLength(line: Int): Int =
val isLastLine = source.lineToOffsetOpt(line).nonEmpty && source.lineToOffsetOpt(line + 1).isEmpty
if isLastLine then source.content.length - source.lineToOffset(line) - 1
else source.lineToOffset(line + 1) - source.lineToOffset(line) - 1 // -1 for newline char
val start = source.lineToOffset(range.startLine) +
math.min(range.startCharacter, lineLength(range.startLine))
val end = source.lineToOffset(range.endLine) +
math.min(range.endCharacter, lineLength(range.endLine))
new String(source.content, start, end - start)
// def pprint(tree: s.Tree, range: Option[Range]): String =
class TreePrinter(source: SourceFile, originalRange: Option[Range], notes: InfoNotes) extends InfoPrinter(notes):
def pprint(tree: Tree): String =
val sb = new StringBuilder()
processTree(tree)(using sb)
sb.toString
private def rep[T](xs: Seq[T], seq: String)(f: T => Unit)(using sb: StringBuilder): Unit =
xs.zipWithIndex.foreach { (x, i) =>
if i != 0 then sb.append(seq)
f(x)
}
private def processTree(tree: Tree)(using sb: StringBuilder): Unit =
tree match {
case tree: ApplyTree =>
processTree(tree.function)
sb.append("(")
rep(tree.arguments, ", ")(processTree)
sb.append(")")
case tree: FunctionTree =>
sb.append("{")
sb.append("(")
rep(tree.parameters, ", ")(processTree)
sb.append(") =>")
processTree(tree.body)
sb.append("}")
case tree: IdTree =>
sb.append(pprintRef(tree.symbol))
case tree: LiteralTree =>
sb.append(pprint(tree.constant))
case tree: MacroExpansionTree =>
sb.append("(`macro-expandee` : `")
sb.append(pprint(tree.tpe))
sb.append(")")
case tree: OriginalTree =>
if (tree.range == originalRange && originalRange.nonEmpty) then
sb.append("*")
else
sb.append("orig(")
sb.append(source.substring(tree.range))
sb.append(")")
case tree: SelectTree =>
processTree(tree.qualifier)
sb.append(".")
tree.id match
case Some(tree) => processTree(tree)
case None => ()
case tree: TypeApplyTree =>
processTree(tree.function)
sb.append("[")
rep(tree.typeArguments, ", ")((t) => sb.append(pprint(t)))
sb.append("]")
case _ =>
sb.append(">")
}
end SyntheticPrinter
© 2015 - 2025 Weber Informatics LLC | Privacy Policy