All Downloads are FREE. Search and download functionalities are using the official Maven repository.

dotty.tools.dotc.reporting.Message.scala Maven / Gradle / Ivy

package dotty.tools
package dotc
package reporting

import core.*
import Contexts.*, Decorators.*, Symbols.*, Types.*, Flags.*
import printing.{RefinedPrinter, MessageLimiter, ErrorMessageLimiter}
import printing.Texts.Text
import printing.Formatting.hl
import config.SourceVersion

import scala.language.unsafeNulls
import scala.annotation.threadUnsafe

/** ## Tips for error message generation
 *
 *  - You can use the `em` interpolator for error messages. It's defined in core.Decorators.
 *  - You can also use a simple string argument for `error` or `warning` (not for the other variants),
 *    but the string should not be interpolated or composed of objects that require a
 *    Context for evaluation.
 *  - When embedding interpolated substrings defined elsewhere in error messages,
 *    use `i` and make sure they are defined as def's instead of vals. That way, the
 *    possibly expensive interpolation will performed only in the case where the message
 *    is eventually printed. Note: At least during typer, it's common for messages
 *    to be discarded without being printed. Also, by making them defs, you ensure that
 *    they will be evaluated in the Message context, which makes formatting safer
 *    and more robust.
 *  - For common messages, or messages that might require explanation, prefer defining
 *    a new `Message` class in file `messages.scala` and use that instead. The advantage is that these
 *    messages have unique IDs that can be referenced elsewhere.
 */
object Message:
  def rewriteNotice(what: String, version: SourceVersion | Null = null, options: String = "")(using Context): String =
    if !ctx.mode.is(Mode.Interactive) then
      val sourceStr = if version != null then i"-source $version" else ""
      val optionStr =
        if options.isEmpty then sourceStr
        else if sourceStr.isEmpty then options
        else i"$sourceStr $options"
      i"\n$what can be rewritten automatically under -rewrite $optionStr."
    else ""

  private type Recorded = Symbol | ParamRef | SkolemType

  private case class SeenKey(str: String, isType: Boolean)

  /** A class that records printed items of one of the types in `Recorded`,
   *  adds superscripts for disambiguations, and can explain recorded symbols
   *  in ` where` clause
   */
  private class Seen(disambiguate: Boolean):

    /** The set of lambdas that were opened at some point during printing. */
    private val openedLambdas = new collection.mutable.HashSet[LambdaType]

    /** Register that `tp` was opened during printing. */
    def openLambda(tp: LambdaType): Unit =
      openedLambdas += tp

    val seen = new collection.mutable.HashMap[SeenKey, List[Recorded]].withDefaultValue(Nil)

    var nonSensical = false

    /** If false, stop all recordings */
    private var recordOK = disambiguate

    /** Clear all entries and stop further entries to be added */
    def disable() =
      seen.clear()
      recordOK = false

    /** Record an entry `entry` with given String representation `str` and a
     *  type/term namespace identified by `isType`.
     *  If the entry was not yet recorded, allocate the next superscript corresponding
     *  to the same string in the same name space. The first recording is the string proper
     *  and following recordings get consecutive superscripts starting with 2.
     *  @return  The possibly superscripted version of `str`.
     */
    def record(str: String, isType: Boolean, entry: Recorded)(using Context): String = if !recordOK then str else
      //println(s"recording $str, $isType, $entry")

      /** If `e1` is an alias of another class of the same name, return the other
       *  class symbol instead. This normalization avoids recording e.g. scala.List
       *  and scala.collection.immutable.List as two different types
       */
      def followAlias(e1: Recorded): Recorded = e1 match {
        case e1: Symbol if e1.isAliasType =>
          val underlying = e1.typeRef.underlyingClassRef(refinementOK = false).typeSymbol
          if (underlying.name == e1.name) underlying else e1.namedType.dealias.typeSymbol
        case _ => e1
      }
      val key = SeenKey(str, isType)
      val existing = seen(key)
      lazy val dealiased = followAlias(entry)

      /** All lambda parameters with the same name are given the same superscript as
       *  long as their corresponding binder has been printed.
       *  See tests/neg/lambda-rename.scala for test cases.
       */
      def sameSuperscript(cur: Recorded, existing: Recorded) =
        (cur eq existing) ||
        (cur, existing).match
          case (cur: ParamRef, existing: ParamRef) =>
            (cur.paramName eq existing.paramName) &&
            openedLambdas.contains(cur.binder) &&
            openedLambdas.contains(existing.binder)
          case _ =>
            false

      // The length of alts corresponds to the number of superscripts we need to print.
      var alts = existing.dropWhile(alt => !sameSuperscript(dealiased, followAlias(alt)))
      if alts.isEmpty then
        alts = entry :: existing
        seen(key) = alts

      val suffix = alts.length match {
        case 1 => ""
        case n => n.toString.toCharArray.map {
          case '0' => '⁰'
          case '1' => '¹'
          case '2' => '²'
          case '3' => '³'
          case '4' => '⁴'
          case '5' => '⁵'
          case '6' => '⁶'
          case '7' => '⁷'
          case '8' => '⁸'
          case '9' => '⁹'
        }.mkString
      }
      str + suffix
    end record

    /** Create explanation for single `Recorded` type or symbol */
    private def explanation(entry: AnyRef)(using Context): String =
      def boundStr(bound: Type, default: ClassSymbol, cmp: String) =
        if (bound.isRef(default)) "" else i"$cmp $bound"

      def boundsStr(bounds: TypeBounds): String = {
        val lo = boundStr(bounds.lo, defn.NothingClass, ">:")
        val hi = boundStr(bounds.hi, defn.AnyClass, "<:")
        if (lo.isEmpty) hi
        else if (hi.isEmpty) lo
        else s"$lo and $hi"
      }

      def addendum(cat: String, info: Type): String = info match {
        case bounds @ TypeBounds(lo, hi) if !(bounds =:= TypeBounds.empty) && !bounds.isErroneous =>
          if (lo eq hi) i" which is an alias of $lo"
          else i" with $cat ${boundsStr(bounds)}"
        case _ =>
          ""
      }

      entry match {
        case param: TypeParamRef =>
          s"is a type variable${addendum("constraint", TypeComparer.bounds(param))}"
        case param: TermParamRef =>
          s"is a reference to a value parameter"
        case sym: Symbol =>
          val info =
            if (ctx.gadt.contains(sym))
              sym.info & ctx.gadt.fullBounds(sym)
            else
              sym.info
          s"is a ${ctx.printer.kindString(sym)}${sym.showExtendedLocation}${addendum("bounds", info)}"
        case tp: SkolemType =>
          s"is an unknown value of type ${tp.widen.show}"
      }
    end explanation

    /** Produce a where clause with explanations for recorded iterms.
     */
    def explanations(using Context): String =
      def needsExplanation(entry: Recorded) = entry match {
        case param: TypeParamRef => ctx.typerState.constraint.contains(param)
        case param: ParamRef     => false
        case skolem: SkolemType  => true
        case sym: Symbol         => ctx.gadt.contains(sym) && ctx.gadt.fullBounds(sym) != TypeBounds.empty
      }

      val toExplain: List[(String, Recorded)] = seen.toList.flatMap { kvs =>
        val res: List[(String, Recorded)] = kvs match {
          case (key, entry :: Nil) =>
            if (needsExplanation(entry)) (key.str, entry) :: Nil else Nil
          case (key, entries) =>
            for (alt <- entries) yield {
              val tickedString = record(key.str, key.isType, alt)
              (tickedString, alt)
            }
        }
        res // help the inferencer out
      }.sortBy(_._1)

      def columnar(parts: List[(String, String)]): List[String] = {
        lazy val maxLen = parts.map(_._1.length).max
        parts.map {
          case (leader, trailer) =>
            val variable = hl(leader)
            s"""$variable${" " * (maxLen - leader.length)} $trailer"""
        }
      }

      val explainParts = toExplain.map { case (str, entry) => (str, explanation(entry)) }
      val explainLines = columnar(explainParts)
      if (explainLines.isEmpty) "" else i"where:    $explainLines%\n          %\n"
    end explanations
  end Seen

  /** Printer to be used when formatting messages */
  private class Printer(val seen: Seen, _ctx: Context) extends RefinedPrinter(_ctx):

    /** True if printer should a show source module instead of its module class */
    private def useSourceModule(sym: Symbol): Boolean =
      sym.is(ModuleClass, butNot = Package) && sym.sourceModule.exists && !_ctx.settings.YdebugNames.value

    override def simpleNameString(sym: Symbol): String =
      if useSourceModule(sym) then simpleNameString(sym.sourceModule)
      else seen.record(super.simpleNameString(sym), sym.isType, sym)

    override def ParamRefNameString(param: ParamRef): String =
      seen.record(super.ParamRefNameString(param), param.isInstanceOf[TypeParamRef], param)

    override def toTextRef(tp: SingletonType): Text = tp match
      case tp: SkolemType => seen.record(tp.repr.toString, isType = true, tp)
      case _ => super.toTextRef(tp)

    override def toTextMethodAsFunction(info: Type, isPure: Boolean, refs: Text): Text =
      info match
        case info: LambdaType =>
          seen.openLambda(info)
        case _ =>
      super.toTextMethodAsFunction(info, isPure, refs)

    override def toText(tp: Type): Text =
      if !tp.exists || tp.isErroneous then seen.nonSensical = true
      tp match
        case tp: TypeRef if useSourceModule(tp.symbol) => Str("object ") ~ super.toText(tp)
        case tp: LambdaType =>
          seen.openLambda(tp)
          super.toText(tp)
        case _ => super.toText(tp)

    override def toText(sym: Symbol): Text =
      sym.infoOrCompleter match
        case _: ErrorType | TypeAlias(_: ErrorType) | NoType => seen.nonSensical = true
        case _ =>
      super.toText(sym)
  end Printer

end Message

/** A `Message` contains all semantic information necessary to easily
  * comprehend what caused the message to be logged. Each message can be turned
  * into a `Diagnostic` which contains the log level and can later be
  * consumed by a subclass of `Reporter`. However, the error position is only
  * part of `Diagnostic`, not `Message`.
  *
  * NOTE: you should not persist a message directly, because most messages take
  * an implicit `Context` and these contexts weigh in at about 4mb per instance.
  * Therefore, persisting these will result in a memory leak.
  *
  * Instead use the `persist` method to create an instance that does not keep a
  * reference to these contexts.
  *
  * @param errorId a unique id identifying the message, this will be
  *                used to reference documentation online
  *
  * Messages modify the rendendering of interpolated strings in several ways:
  *
  *  1. The size of the printed code is limited with a MessageLimiter. If the message
  *    would get too large or too deeply nested, a `...` is printed instead.
  *  2. References to module classes are prefixed with `object` for better recognizability.
  *  3. A where clause is sometimes added which contains the following additional explanations:
  *     - References are disambiguated: If a message contains occurrences of the same identifier
  *       representing different symbols, the duplicates are printed with superscripts
  *       and the where-clause explains where each symbol is located.
  *     - Uninstantiated variables are explained in the where-clause with additional
  *       info about their bounds.
  *     - Skolems are explained with additional info about their underlying type.
  *
  *  Messages inheriting from the NoDisambiguation trait or returned from the
  *  `noDisambiguation()` method skip point (3) above. This makes sense if the
  *  message already exolains where different occurrences of the same identifier
  *  are located. Examples are NamingMsgs such as double definition errors,
  *  overriding errors, and ambiguous implicit errors.
  *
  *  We consciously made the design decision to disambiguate by default and disable
  *  disambiguation as an opt-in. The reason is that one usually does not consider all
  *  fine-grained details when writing an error message. If disambiguation is the default,
  *  some tests will show where clauses that look too noisy and that then can be disabled
  *  when needed. But if silence is the default, one usually does not realize that
  *  better info could be obtained by turning disambiguation on.
  */
abstract class Message(val errorId: ErrorMessageID)(using Context) { self =>
  import Message.*

  /** The kind of the error message, e.g. "Syntax" or "Type Mismatch".
    * This will be printed as "$kind Error", "$kind Warning", etc, on the first
    * line of the message.
    */
  def kind: MessageKind

  /** The `msg` contains the diagnostic message e.g:
    *
    * > expected: String
    * > found:    Int
    *
    * This message will be placed underneath the position given by the enclosing
    * `Diagnostic`. The message is given in raw form, with possible embedded
    *   tags.
    */
  protected def msg(using Context): String

  /** The explanation should provide a detailed description of why the error
    * occurred and use examples from the user's own code to illustrate how to
    * avoid these errors. It might contain embedded  tags.
    */
  protected def explain(using Context): String

  /** What gets printed after the message proper */
  protected def msgPostscript(using Context): String =
    if ctx eq NoContext then ""
    else ctx.printer match
      case msgPrinter: Message.Printer =>
        myIsNonSensical = msgPrinter.seen.nonSensical
        val addendum = msgPrinter.seen.explanations
        msgPrinter.seen.disable()
          // Clear entries and stop futher recording so that messages containing the current
          // one don't repeat the explanations or use explanations from the msgPostscript.
        if addendum.isEmpty then "" else "\n\n" ++ addendum
      case _ =>
        ""

  /** Does this message have an explanation?
   *  This is normally the same as `explain.nonEmpty` but can be overridden
   *  if we need a way to return `true` without actually calling the
   *  `explain` method.
   */
  def canExplain: Boolean = explain.nonEmpty

  private var myIsNonSensical: Boolean = false

  /** A message is non-sensical if it contains references to internally
   *  generated error types. Normally we want to suppress error messages
   *  referring to types like this because they look weird and are normally
   *  follow-up errors to something that was diagnosed before.
   */
  def isNonSensical: Boolean = { message; myIsNonSensical }

  private var disambiguate: Boolean = true

  def withoutDisambiguation(): this.type =
    disambiguate = false
    this

  private def inMessageContext(disambiguate: Boolean)(op: Context ?=> String): String =
    if ctx eq NoContext then op
    else
      val msgContext = ctx.printer match
        case _: Message.Printer => ctx
        case _ =>
          val seen = Seen(disambiguate)
          val ctx1 = ctx.fresh.setPrinterFn(Message.Printer(seen, _))
          if !ctx1.property(MessageLimiter).isDefined then
            ctx1.setProperty(MessageLimiter, ErrorMessageLimiter())
          ctx1
      op(using msgContext)

  /** The message to report.  tags are filtered out */
  @threadUnsafe lazy val message: String =
    inMessageContext(disambiguate)(msg + msgPostscript)

  /** The explanation to report.  tags are filtered out */
  @threadUnsafe lazy val explanation: String =
    inMessageContext(disambiguate = false)(explain)

  /** The implicit `Context` in messages is a large thing that we don't want
    * persisted. This method gets around that by duplicating the message,
    * forcing its `msg` and `explanation` vals and dropping the implicit context
    * that was captured in the original message.
    */
  def persist: Message = new Message(errorId)(using NoContext):
    val kind  = self.kind
    private val persistedMsg = self.message
    private val persistedExplain = self.explanation
    def msg(using Context) = persistedMsg
    def explain(using Context) = persistedExplain
    override val canExplain = self.canExplain
    override def isNonSensical = self.isNonSensical

  def append(suffix: => String): Message = mapMsg(_ ++ suffix)
  def prepend(prefix: => String): Message = mapMsg(prefix ++ _)

  def mapMsg(f: String => String): Message = new Message(errorId):
    val kind = self.kind
    def msg(using Context) = f(self.msg)
    def explain(using Context) = self.explain
    override def canExplain = self.canExplain

  def appendExplanation(suffix: => String): Message = new Message(errorId):
    val kind = self.kind
    def msg(using Context) = self.msg
    def explain(using Context) = self.explain ++ suffix
    override def canExplain = true

  /** Override with `true` for messages that should always be shown even if their
   *  position overlaps another message of a different class. On the other hand
   *  multiple messages of the same class with overlapping positions will lead
   *  to only a single message of that class to be issued.
   */
  def showAlways = false

  /** A list of actions attached to this message to address the issue this
    * message represents.
    */
  def actions(using Context): List[CodeAction] = List.empty

  override def toString = msg
}

/** A marker trait that suppresses generation of `where` clause for disambiguations */
trait NoDisambiguation extends Message:
  withoutDisambiguation()

/** The fallback `Message` containing no explanation and having no `kind` */
final class NoExplanation(msgFn: Context ?=> String)(using Context) extends Message(ErrorMessageID.NoExplanationID) {
  def msg(using Context): String = msgFn
  def explain(using Context): String = ""
  val kind: MessageKind = MessageKind.NoKind

  override def toString(): String = msg
}

/** The extractor for `NoExplanation` can be used to check whether any error
  * lacks an explanation
  */
object NoExplanation {
  def unapply(m: Message): Option[Message] =
    if (m.explanation == "") Some(m)
    else None
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy