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

dotty.tools.languageserver.DottyLanguageServer.scala Maven / Gradle / Ivy

There is a newer version: 3.6.4-RC1-bin-20241220-0bfa1af-NIGHTLY
Show newest version
package dotty.tools
package languageserver

import java.net.URI
import java.io._
import java.nio.file._
import java.util.concurrent.{CompletableFuture, ConcurrentHashMap}
import java.util.function.Function

import com.fasterxml.jackson.databind.ObjectMapper

import org.eclipse.lsp4j

import scala.collection._
import scala.jdk.CollectionConverters._
import scala.util.control.NonFatal
import scala.io.Codec

import dotc._
import ast.{Trees, tpd, untpd}
import core._, core.Decorators._
import Comments._, Constants._, Contexts._, Flags._, Names._, NameOps._, Symbols._, SymDenotations._, Trees._, Types._
import classpath.ClassPathEntries
import reporting._
import typer.Typer
import util._
import interactive._, interactive.InteractiveDriver._
import decompiler.IDEDecompilerDriver
import Interactive.Include
import config.Printers.interactiv

import languageserver.config.ProjectConfig
import languageserver.worksheet.{Worksheet, WorksheetService}
import languageserver.decompiler.{TastyDecompilerService}

import lsp4j.services._

/** An implementation of the Language Server Protocol for Dotty.
 *
 *  You should not have to directly this class, instead see `dotty.tools.languageserver.Main`.
 *
 *  For more information see:
 *  - The LSP is defined at https://github.com/Microsoft/language-server-protocol
 *  - This implementation is based on the LSP4J library: https://github.com/eclipse/lsp4j
 */
class DottyLanguageServer extends LanguageServer
    with TextDocumentService with WorkspaceService with WorksheetService with TastyDecompilerService { thisServer =>
  import ast.tpd._

  import DottyLanguageServer._
  import InteractiveDriver._

  import lsp4j.jsonrpc.{CancelChecker, CompletableFutures}
  import lsp4j.jsonrpc.messages.{Either => JEither}
  import lsp4j._

  private[this] var rootUri: String = _

  private[this] var myClient: DottyClient = _
  def client: DottyClient = myClient

  private[this] var myDrivers: mutable.Map[ProjectConfig, InteractiveDriver] = _

  private[this] var myDependentProjects: mutable.Map[ProjectConfig, mutable.Set[ProjectConfig]] = _

  def drivers: Map[ProjectConfig, InteractiveDriver] = thisServer.synchronized {
    if myDrivers == null then
      assert(rootUri != null, "`drivers` cannot be called before `initialize`")
      val configFile = new File(new URI(rootUri + '/' + IDE_CONFIG_FILE))
      val configs: List[ProjectConfig] = (new ObjectMapper).readValue(configFile, classOf[Array[ProjectConfig]]).toList

      val defaultFlags = List("-color:never" /*, "-Yplain-printer","-Yprint-pos"*/)

      myDrivers = new mutable.HashMap
      for (config <- configs) {
        implicit class updateDeco(ss: List[String]) {
          def update(pathKind: String, pathInfo: String) = {
            val idx = ss.indexOf(pathKind)
            val ss1 = if idx >= 0 then ss.take(idx) ++ ss.drop(idx + 2) else ss
            ss1 ++ List(pathKind, pathInfo)
          }
        }
        val settings =
          defaultFlags ++
          config.compilerArguments.toList
            .update("-d", config.classDirectory.getAbsolutePath)
            .update("-classpath", (config.classDirectory +: config.dependencyClasspath).mkString(File.pathSeparator))
            .update("-sourcepath", config.sourceDirectories.mkString(File.pathSeparator))
        myDrivers(config) = new InteractiveDriver(settings)
      }
    myDrivers
  }

  /** Restart all presentation compiler drivers, copying open files over */
  private def restart() = thisServer.synchronized {
    interactiv.println("restarting presentation compiler")
    val driverConfigs = for ((config, driver) <- myDrivers.toList) yield
      (config, new InteractiveDriver(driver.settings), driver.openedFiles)
    for ((config, driver, _) <- driverConfigs)
      myDrivers(config) = driver
    System.gc()
    for ((_, driver, opened) <- driverConfigs; (uri, source) <- opened)
      driver.run(uri, source)
    if Memory.isCritical() then
      println(s"WARNING: Insufficient memory to run Scala language server on these projects.")
  }

  private def checkMemory() =
    if Memory.isCritical() then
      CompletableFutures.computeAsync { _ => restart() }

  /** The configuration of the project that owns `uri`. */
  def configFor(uri: URI): ProjectConfig = thisServer.synchronized {
    val config =
      drivers.keys.find(config => config.sourceDirectories.exists(sourceDir =>
        new File(uri.getPath).getCanonicalPath.startsWith(sourceDir.getCanonicalPath)))

    config.getOrElse {
      val config = drivers.keys.head
      // println(s"No configuration contains $uri as a source file, arbitrarily choosing ${config.id}")
      config
    }
  }

  /** The driver instance responsible for compiling `uri` */
  def driverFor(uri: URI): InteractiveDriver = {
    drivers(configFor(uri))
  }

    /** The driver instance responsible for decompiling `uri` in `classPath` */
  def decompilerDriverFor(uri: URI, classPath: String): IDEDecompilerDriver = thisServer.synchronized {
    val config = configFor(uri)
    val defaultFlags = List("-color:never")

    implicit class updateDeco(ss: List[String]) {
      def update(pathKind: String, pathInfo: String) = {
        val idx = ss.indexOf(pathKind)
        val ss1 = if idx >= 0 then ss.take(idx) ++ ss.drop(idx + 2) else ss
        ss1 ++ List(pathKind, pathInfo)
      }
    }
    val settings =
      defaultFlags ++
      config.compilerArguments.toList
        .update("-classpath", (classPath +: config.dependencyClasspath).mkString(File.pathSeparator))
    new IDEDecompilerDriver(settings)
  }

  /** A mapping from project `p` to the set of projects that transitively depend on `p`. */
  def dependentProjects: Map[ProjectConfig, Set[ProjectConfig]] = thisServer.synchronized {
    if myDependentProjects == null then
      val idToConfig = drivers.keys.map(k => k.id -> k).toMap
      val allProjects = drivers.keySet

      def transitiveDependencies(config: ProjectConfig): Set[ProjectConfig] = {
        val dependencies = config.projectDependencies.flatMap(idToConfig.get).toSet
        dependencies ++ dependencies.flatMap(transitiveDependencies)
      }

      myDependentProjects = new mutable.HashMap().withDefaultValue(mutable.Set.empty)
      for { project <- allProjects
            dependency <- transitiveDependencies(project) } {
        myDependentProjects(dependency) += project
      }
    myDependentProjects
  }

  def connect(client: DottyClient): Unit = {
    myClient = client
  }

  override def exit(): Unit = {
    System.exit(0)
  }

  override def shutdown(): CompletableFuture[Object] = {
    CompletableFuture.completedFuture(new Object)
  }

  def computeAsync[R](fun: CancelChecker => R, synchronize: Boolean = true): CompletableFuture[R] =
    CompletableFutures.computeAsync { cancelToken =>
      // We do not support any concurrent use of the compiler currently.
      def computation(): R = {
        cancelToken.checkCanceled()
        checkMemory()
        try {
          fun(cancelToken)
        } catch {
          case NonFatal(ex) =>
            ex.printStackTrace
            throw ex
        }
      }
      if synchronize then
        thisServer.synchronized { computation() }
      else
        computation()
    }

  override def initialize(params: InitializeParams) = computeAsync { cancelToken =>
    rootUri = params.getRootUri
    assert(rootUri != null)

    class DottyServerCapabilities(val worksheetRunProvider: Boolean = true,
                                  val tastyDecompiler: Boolean = true) extends lsp4j.ServerCapabilities

    val c = new DottyServerCapabilities
    c.setTextDocumentSync(TextDocumentSyncKind.Full)
    c.setDocumentHighlightProvider(true)
    c.setDocumentSymbolProvider(true)
    c.setDefinitionProvider(true)
    c.setRenameProvider(true)
    c.setHoverProvider(true)
    c.setWorkspaceSymbolProvider(true)
    c.setReferencesProvider(true)
    c.setImplementationProvider(true)
    c.setCompletionProvider(new CompletionOptions(
      /* resolveProvider = */ false,
      /* triggerCharacters = */ List(".").asJava))
    c.setSignatureHelpProvider(new SignatureHelpOptions(
      /* triggerCharacters = */ List("(").asJava))

    // Do most of the initialization asynchronously so that we can return early
    // from this method and thus let the client know our capabilities.
    CompletableFuture.supplyAsync(() => drivers)
      .exceptionally { (ex: Throwable) =>
        ex.printStackTrace
        sys.exit(1)
      }

    new InitializeResult(c)
  }

  override def didOpen(params: DidOpenTextDocumentParams): Unit = thisServer.synchronized {
    checkMemory()
    val document = params.getTextDocument
    val uri = new URI(document.getUri)
    val driver = driverFor(uri)
    implicit def ctx: Context = driver.currentCtx
    val worksheetMode = isWorksheet(uri)

    val text =
      if (worksheetMode)
        wrapWorksheet(document.getText)
      else
        document.getText

    val diags = driver.run(uri, text)

    client.publishDiagnostics(new PublishDiagnosticsParams(
      document.getUri,
      diags.flatMap(diagnostic).asJava))
  }

  override def didChange(params: DidChangeTextDocumentParams): Unit = {
    val document = params.getTextDocument
    val uri = new URI(document.getUri)
    val worksheetMode = isWorksheet(uri)

    thisServer.synchronized  {
      checkMemory()

      val driver = driverFor(uri)
      implicit def ctx: Context = driver.currentCtx

      val change = params.getContentChanges.get(0)
      assert(change.getRange == null, "TextDocumentSyncKind.Incremental support is not implemented")

      val text =
        if (worksheetMode)
          wrapWorksheet(change.getText)
        else
          change.getText

      val diags = driver.run(uri, text)

      client.publishDiagnostics(new PublishDiagnosticsParams(
        document.getUri,
        diags.flatMap(diagnostic).asJava))
    }
  }

  override def didClose(params: DidCloseTextDocumentParams): Unit = thisServer.synchronized {
    val document = params.getTextDocument
    val uri = new URI(document.getUri)

    driverFor(uri).close(uri)
  }

  override def didChangeConfiguration(params: DidChangeConfigurationParams): Unit =
    /*thisServer.synchronized*/ {}

  override def didChangeWatchedFiles(params: DidChangeWatchedFilesParams): Unit =
    /*thisServer.synchronized*/ {}

  override def didSave(params: DidSaveTextDocumentParams): Unit = {
    /*thisServer.synchronized*/ {}
  }

  // FIXME: share code with NotAMember
  override def completion(params: CompletionParams) = computeAsync { cancelToken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)
    implicit def ctx: Context = driver.currentCtx

    val pos = sourcePosition(driver, uri, params.getPosition)
    val items = driver.compilationUnits.get(uri) match {
      case Some(unit) =>
        val freshCtx = ctx.fresh.setCompilationUnit(unit)
        Completion.completions(pos)(using freshCtx)._2
      case None => Nil
    }

    JEither.forRight(new CompletionList(
      /*isIncomplete = */ false, items.map(completionItem).asJava))
  }

  /** If cursor is on a reference, show its definition and all overriding definitions.
   *  If cursor is on a definition, show this definition together with all overridden
   *  and overriding definitions.
   */
  override def definition(params: TextDocumentPositionParams) = computeAsync { cancelToken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)
    implicit def ctx: Context = driver.currentCtx

    val pos = sourcePosition(driver, uri, params.getPosition)
    val path = Interactive.pathTo(driver.openedTrees(uri), pos)

    val definitions = Interactive.findDefinitions(path, pos, driver).toList
    definitions.flatMap(d => location(d.namePos)).asJava
  }

  override def references(params: ReferenceParams) = computeAsync { cancelToken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)

    val includes = {
      val includeDeclaration = params.getContext.isIncludeDeclaration
      Include.references | Include.overriding | Include.imports | Include.local |
        (if (includeDeclaration) Include.definitions else Include.empty)
    }

    val uriTrees = driver.openedTrees(uri)
    val pos = sourcePosition(driver, uri, params.getPosition)

    val (definitions, originalSymbols) = {
      implicit def ctx: Context = driver.currentCtx
      val path = Interactive.pathTo(driver.openedTrees(uri), pos)
      val definitions = Interactive.findDefinitions(path, pos, driver)
      val originalSymbols = Interactive.enclosingSourceSymbols(path, pos)

      (definitions, originalSymbols)
    }

    val references = {
      // Collect the information necessary to look into each project separately: representation of
      // `originalSymbol` in this project, the context and correct Driver.
      val perProjectInfo = inProjectsSeeing(driver, definitions, originalSymbols)

      perProjectInfo.flatMap { (remoteDriver, ctx, definitions) =>
        definitions.flatMap { definition =>
          val name = definition.name(using ctx).sourceModuleName.toString
          val trees = remoteDriver.sourceTreesContaining(name)(using ctx)
          val matches = Interactive.findTreesMatching(trees, includes, definition)(using ctx)
          matches.map(tree => location(tree.namePos(using ctx)))
        }
      }
    }.toList

    references.flatten.distinct.asJava
  }

  override def rename(params: RenameParams) = computeAsync { cancelToken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)
    implicit def ctx: Context = driver.currentCtx

    val uriTrees = driver.openedTrees(uri)
    val pos = sourcePosition(driver, uri, params.getPosition)
    val path = Interactive.pathTo(uriTrees, pos)
    val syms = Interactive.enclosingSourceSymbols(path, pos)
    val newName = params.getNewName

    def findRenamedReferences(trees: List[SourceTree], syms: List[Symbol], withName: Name): List[SourceTree] = {
      val includes = Include.all
      syms.flatMap { sym =>
        Interactive.findTreesMatching(trees, Include.all, sym, t => Interactive.sameName(t.name, withName))
      }
    }

    val refs =
      path match {
        // Selected a renaming in an import node
        case untpd.ImportSelector(_, rename: untpd.Ident, _) :: (_: Import) :: rest if rename.span.contains(pos.span) =>
          findRenamedReferences(uriTrees, syms, rename.name)

        // Selected a reference that has been renamed
        case (nameTree: NameTree) :: rest if Interactive.isRenamed(nameTree) =>
          findRenamedReferences(uriTrees, syms, nameTree.name)

        case _ =>
          val (include, allSymbols) =
            if (syms.exists(_.allOverriddenSymbols.nonEmpty)) {
              showMessageRequest(MessageType.Info,
                RENAME_OVERRIDDEN_QUESTION,
                List(
                  RENAME_OVERRIDDEN    -> (() => (Include.all, syms.flatMap(s => s :: s.allOverriddenSymbols.toList))),
                  RENAME_NO_OVERRIDDEN -> (() => (Include.all.except(Include.overridden), syms)))
              ).get.getOrElse((Include.empty, Nil))
            } else {
              (Include.all, syms)
            }

          val names = allSymbols.map(_.name.sourceModuleName).toSet
          val definitions = Interactive.findDefinitions(allSymbols, driver, include.isOverridden, includeExternal = true)
          val perProjectInfo = inProjectsSeeing(driver, definitions, allSymbols)

          perProjectInfo.flatMap { (remoteDriver, ctx, definitions) =>
            definitions.flatMap { definition =>
              val name = definition.name(using ctx).sourceModuleName.toString
              val trees = remoteDriver.sourceTreesContaining(name)(using ctx)
              Interactive.findTreesMatching(trees,
                                            include,
                                            definition,
                                            t => names.exists(Interactive.sameName(t.name, _)))(using ctx)
            }
          }
      }

    val changes =
      refs.groupBy(ref => toUriOption(ref.source))
        .flatMap { case (uriOpt, refs) => uriOpt.map(uri => (uri.toString, refs)) }
        .transform((_, refs) => refs.flatMap(ref =>
          range(ref.namePos).map(nameRange => new TextEdit(nameRange, newName))).distinct.asJava)

    new WorkspaceEdit(changes.asJava)
  }

  override def documentHighlight(params: TextDocumentPositionParams) = computeAsync { cancelToken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)
    implicit def ctx: Context = driver.currentCtx

    val pos = sourcePosition(driver, uri, params.getPosition)
    val uriTrees = driver.openedTrees(uri)
    val path = Interactive.pathTo(uriTrees, pos)
    val syms = Interactive.enclosingSourceSymbols(path, pos)
    val includes = Include.all.except(Include.linkedClass)

    syms.flatMap { sym =>
      val refs = Interactive.findTreesMatching(uriTrees, includes, sym)
      (for {
        ref <- refs
        nameRange <- range(ref.namePos)
      } yield new DocumentHighlight(nameRange, DocumentHighlightKind.Read))
    }.distinct.asJava
  }

  override def hover(params: TextDocumentPositionParams) = computeAsync { cancelToken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)
    implicit def ctx: Context = driver.currentCtx

    val pos = sourcePosition(driver, uri, params.getPosition)
    val trees = driver.openedTrees(uri)
    val path = Interactive.pathTo(trees, pos)
    val tp = Interactive.enclosingType(trees, pos)
    val tpw = tp.widenTermRefExpr

    if (tp.isError || tpw == NoType) null // null here indicates that no response should be sent
    else {
      Interactive.enclosingSourceSymbols(path, pos) match {
        case Nil =>
          null
        case symbols =>
          val docComments = symbols.flatMap(ParsedComment.docOf)
          val content = hoverContent(Some(tpw.show), docComments)
          new Hover(content, null)
      }
    }
  }

  override def documentSymbol(params: DocumentSymbolParams) = computeAsync { cancelToken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)
    implicit def ctx: Context = driver.currentCtx

    val uriTrees = driver.openedTrees(uri)

    // Excludes type and term params from synthetic symbols
    val excludeParamsFromSyntheticSymbols = (n: NameTree) => {
      val owner = n.symbol.owner
      !n.symbol.is(Param) || {
        !owner.is(Synthetic) &&
        !owner.isPrimaryConstructor
      }
    }

    val defs = Interactive.namedTrees(uriTrees, Include.local, excludeParamsFromSyntheticSymbols)

    (for {
      d <- defs if (!isWorksheetWrapper(d) && !isTopLevelWrapper(d))
      info <- symbolInfo(d.tree.symbol, d.pos)
    } yield JEither.forLeft(info)).asJava
  }

  override def symbol(params: WorkspaceSymbolParams) = computeAsync { cancelToken =>
    val query = params.getQuery

    drivers.values.toList.flatMap { driver =>
      implicit def ctx: Context = driver.currentCtx

      val trees = driver.sourceTreesContaining(query)
      val defs = Interactive.namedTrees(trees, Include.empty, _.name.toString.contains(query))
      defs.flatMap(d => symbolInfo(d.tree.symbol, d.namePos))
    }.asJava
  }

  override def implementation(params: TextDocumentPositionParams) = computeAsync { cancelToken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)

    val pos = sourcePosition(driver, uri, params.getPosition)

    val (definitions, originalSymbols) = {
      implicit def ctx: Context = driver.currentCtx
      val path = Interactive.pathTo(driver.openedTrees(uri), pos)
      val originalSymbols = Interactive.enclosingSourceSymbols(path, pos)
      val definitions = Interactive.findDefinitions(path, pos, driver)
      (definitions, originalSymbols)
    }

    val implementations = {
      val perProjectInfo = inProjectsSeeing(driver, definitions, originalSymbols)

      perProjectInfo.flatMap { (remoteDriver, ctx, definitions) =>
        val trees = remoteDriver.sourceTrees(using ctx)
        val predicate: NameTree => Boolean = {
          val predicates = definitions.map(Interactive.implementationFilter(_)(using ctx))
          tree => predicates.exists(_(tree))
        }
        val matches = Interactive.namedTrees(trees, Include.local, predicate)(using ctx)
        matches.map(tree => location(tree.namePos(using ctx)))
      }
    }.toList

    implementations.flatten.asJava
  }

  override def signatureHelp(params: TextDocumentPositionParams) = computeAsync { canceltoken =>
    val uri = new URI(params.getTextDocument.getUri)
    val driver = driverFor(uri)
    implicit def ctx: Context = driver.currentCtx

    val pos = sourcePosition(driver, uri, params.getPosition)
    val trees = driver.openedTrees(uri)
    val path = Interactive.pathTo(trees, pos)
    val (paramN, callableN, signatures) = Signatures.signatureHelp(path, pos.span)
    new SignatureHelp(signatures.map(signatureToSignatureInformation).asJava, callableN, paramN)

  }

  override def getTextDocumentService: TextDocumentService = this
  override def getWorkspaceService: WorkspaceService = this

  // Unimplemented features. If you implement one of them, you may need to add a
  // capability in `initialize`
  override def codeAction(params: CodeActionParams) = null
  override def codeLens(params: CodeLensParams) = null
  override def formatting(params: DocumentFormattingParams) = null
  override def rangeFormatting(params: DocumentRangeFormattingParams) = null
  override def onTypeFormatting(params: DocumentOnTypeFormattingParams) = null
  override def resolveCodeLens(params: CodeLens) = null
  override def resolveCompletionItem(params: CompletionItem) = null

  /**
   * Find the set of projects that have any of `definitions` on their classpath.
   *
   * @param definitions The definitions to consider when looking for projects.
   * @return The set of projects that have any of `definitions` on their classpath.
   */
  private def projectsSeeing(definitions: List[SourceTree])(implicit ctx: Context): Set[ProjectConfig] = {
    if (definitions.isEmpty) {
      drivers.keySet
    } else {
      for {
        definition <- definitions.toSet
        uri <- toUriOption(definition.pos.source).toSet
        config = configFor(uri)
        project <- dependentProjects(config) union Set(config)
      } yield project
    }
  }

  /**
   * Finds projects that can see any of `definitions`, translate `symbols` in their universe.
   *
   * @param baseDriver  The driver responsible for the trees in `definitions` and `symbol`.
   * @param definitions The definitions to consider when looking for projects.
   * @param symbol      Symbols to translate in the universes of the remote projects.
   * @return A list consisting of the remote drivers, their context, and the translation of `symbol`
   *         into their universe.
   */
  private def inProjectsSeeing(baseDriver: InteractiveDriver,
                               definitions: List[SourceTree],
                               symbols: List[Symbol]): List[(InteractiveDriver, Context, List[Symbol])] = {
    val projects = projectsSeeing(definitions)(baseDriver.currentCtx)
    projects.toList.map { config =>
      val remoteDriver = drivers(config)
      val ctx = remoteDriver.currentCtx
      val definitions = symbols.map(Interactive.localize(_, baseDriver, remoteDriver))
      (remoteDriver, ctx, definitions)
    }
  }

  /**
   * Send a `window/showMessageRequest` to the client, asking to choose between `choices`, and
   * perform the associated operation.
   *
   * @param tpe     The type of the request
   * @param message The message accompanying the request
   * @param choices The choices and their associated operation
   * @return A future that will complete with the result of executing the action corresponding to
   *         the user's response.
   */
  private def showMessageRequest[T](tpe: MessageType,
                                    message: String,
                                    choices: List[(String, () => T)]): CompletableFuture[Option[T]] = {
    val options = choices.map((title, _) => new MessageActionItem(title))
    val request = new ShowMessageRequestParams(options.asJava)
    request.setMessage(message)
    request.setType(tpe)

    client.showMessageRequest(request).thenApply { (message: MessageActionItem) =>
      for {
        answer <- Option(message)
        (_, action) <- choices.find(_._1 == answer.getTitle)
      } yield action()
    }
  }
}

object DottyLanguageServer {

  final val IDE_CONFIG_FILE = ".dotty-ide.json"

  final val RENAME_OVERRIDDEN_QUESTION = "Do you want to rename the base member, or only this member?"
  final val RENAME_OVERRIDDEN= "Rename the base member"
  final val RENAME_NO_OVERRIDDEN = "Rename only this member"

  /** Convert an lsp4j.Position to a SourcePosition */
  def sourcePosition(driver: InteractiveDriver, uri: URI, pos: lsp4j.Position): SourcePosition = {
    val actualPosition =
      if (isWorksheet(uri)) toWrappedPosition(pos)
      else pos
    val source = driver.openedFiles(uri)
    if (source.exists)
      source.lineToOffsetOpt(actualPosition.getLine).map(_ + actualPosition.getCharacter) match {
        // `<=` to allow an offset to point to the end of the file
        case Some(offset) if offset <= source.content().length =>
          val p = Spans.Span(offset)
          new SourcePosition(source, p)
        case _ =>
          NoSourcePosition
      }
    else NoSourcePosition
  }

  /** Convert a SourcePosition to an lsp4j.Range */
  def range(p: SourcePosition): Option[lsp4j.Range] =
    if (p.exists) {
      val mappedPosition = positionMapperFor(p.source).map(_(p)).getOrElse(p)
      Some(new lsp4j.Range(
        new lsp4j.Position(mappedPosition.startLine, mappedPosition.startColumn),
        new lsp4j.Position(mappedPosition.endLine, mappedPosition.endColumn)
      ))
    } else
      None

  /** Convert a SourcePosition to an lsp4.Location */
  def location(p: SourcePosition): Option[lsp4j.Location] =
    for {
      uri <- toUriOption(p.source)
      r <- range(p)
    } yield new lsp4j.Location(uri.toString, r)

  /**
   * Convert a Diagnostic to an lsp4j.Diagnostic.
   */
  def diagnostic(dia: Diagnostic)(implicit ctx: Context): Option[lsp4j.Diagnostic] =
    if (!dia.pos.exists)
      None // diagnostics without positions are not supported: https://github.com/Microsoft/language-server-protocol/issues/249
    else {
      def severity(level: Int): lsp4j.DiagnosticSeverity = {
        import interfaces.{Diagnostic => D}
        import lsp4j.{DiagnosticSeverity => DS}

        level match {
          case D.INFO =>
            DS.Information
          case D.WARNING =>
            DS.Warning
          case D.ERROR =>
            DS.Error
        }
      }

      val message = dia.msg
      if (displayMessage(message, dia.pos.source)) {
        val code = message.errorId.errorNumber.toString
        range(dia.pos).map(r =>
            new lsp4j.Diagnostic(
              r, dia.message, severity(dia.level), /*source =*/ "", code))
      } else {
        None
      }
    }

  /**
   * Check whether `message` should be displayed in the IDE.
   *
   * Currently we only filter out the warning about pure expressions in statement position when they
   * are immediate children of the worksheet wrapper.
   *
   * @param message    The message to filter.
   * @param sourceFile The sourcefile from which `message` originates.
   * @return true if the message should be displayed in the IDE, false otherwise.
   */
  private def displayMessage(message: Message, sourceFile: SourceFile)(implicit ctx: Context): Boolean = {
    if (isWorksheet(sourceFile)) {
      message match {
        case msg: PureExpressionInStatementPosition =>
          val ownerSym = if (msg.exprOwner.isLocalDummy) msg.exprOwner.owner else msg.exprOwner
          !isWorksheetWrapper(ownerSym)
        case _ =>
          true
      }
    } else {
      true
    }
  }

  /** Does this URI represent a worksheet? */
  private def isWorksheet(uri: URI): Boolean =
    uri.toString.endsWith(".sc")

  /** Does this sourcefile represent a worksheet? */
  private def isWorksheet(sourcefile: SourceFile): Boolean =
    sourcefile.file.extension == "sc"

  /** Wrap the source of a worksheet inside an `object`. */
  private def wrapWorksheet(source: String): String =
    s"""object ${StdNames.nme.WorksheetWrapper} {
       |$source
       |}""".stripMargin

  /**
   * Map `position` in a wrapped worksheet to the same position in the unwrapped source.
   *
   * Because worksheet are wrapped in an `object`, the positions in the source are one line
   * above from what the compiler sees.
   *
   * @see wrapWorksheet
   * @param position The position as seen by the compiler (after wrapping)
   * @return The position in the actual source file (before wrapping).
   */
  private def toUnwrappedPosition(position: SourcePosition): SourcePosition = {
    new SourcePosition(position.source, position.span, position.outer) {
      override def startLine: Int = position.startLine - 1
      override def endLine: Int = position.endLine - 1
    }
  }

  /**
   * Map `position` in an unwrapped worksheet to the same position in the wrapped source.
   *
   * Because worksheet are wrapped in an `object`, the positions in the source are one line
   * above from what the compiler sees.
   *
   * @see wrapWorksheet
   * @param position The position as seen by VSCode (before wrapping)
   * @return The position as seen by the compiler (after wrapping)
   */
  private def toWrappedPosition(position: lsp4j.Position): lsp4j.Position = {
    new lsp4j.Position(position.getLine + 1, position.getCharacter)
  }

  /**
   * Returns the position mapper necessary to unwrap positions for `sourcefile`. If `sourcefile` is
   * not a worksheet, no mapper is necessary. Otherwise, return `toUnwrappedPosition`.
   */
  private def positionMapperFor(sourcefile: SourceFile): Option[SourcePosition => SourcePosition] = {
    if (isWorksheet(sourcefile)) Some(toUnwrappedPosition _)
    else None
  }

  /**
   * Is `sourceTree` the wrapper object that we put around worksheet sources?
   *
   * @see wrapWorksheet
   */
  def isWorksheetWrapper(sourceTree: SourceTree)(implicit ctx: Context): Boolean = {
    isWorksheet(sourceTree.source) && isWorksheetWrapper(sourceTree.tree.symbol)
  }

  /**
   * Is this symbol the wrapper object that we put around worksheet sources?
   *
   * @see wrapWorksheet
   */
  def isWorksheetWrapper(symbol: Symbol)(implicit ctx: Context): Boolean = {
      symbol.name == StdNames.nme.WorksheetWrapper.moduleClassName &&
      symbol.owner == ctx.definitions.EmptyPackageClass
  }

  /**
   * Is this symbol the wrapper object for top level definitions?
   */
  def isTopLevelWrapper(sourceTree: SourceTree)(implicit ctx: Context): Boolean = {
    val symbol = sourceTree.tree.symbol
    symbol.isPackageObject
  }

  /** Create an lsp4j.CompletionItem from a completion result */
  def completionItem(completion: Completion)(implicit ctx: Context): lsp4j.CompletionItem = {
    def completionItemKind(sym: Symbol)(implicit ctx: Context): lsp4j.CompletionItemKind = {
      import lsp4j.{CompletionItemKind => CIK}

      if sym.is(Package) || sym.is(Module) then
        CIK.Module // No CompletionItemKind.Package (https://github.com/Microsoft/language-server-protocol/issues/155)
      else if sym.isConstructor then
        CIK.Constructor
      else if sym.isClass then
        CIK.Class
      else if sym.is(Mutable) then
        CIK.Variable
      else if sym.is(Method) then
        CIK.Method
      else
        CIK.Field
    }

    val item = new lsp4j.CompletionItem(completion.label)
    item.setDetail(completion.description)

    val documentation = for {
      sym <- completion.symbols
      doc <- ParsedComment.docOf(sym)
    } yield doc

    if (documentation.nonEmpty) {
      item.setDocumentation(hoverContent(None, documentation))
    }

    item.setDeprecated(completion.symbols.forall(_.hasAnnotation(defn.DeprecatedAnnot)))
    completion.symbols.headOption.foreach(s => item.setKind(completionItemKind(s)))
    item
  }

  def markupContent(content: String): lsp4j.MarkupContent = {
    if content.isEmpty then
      null
    else {
      val markup = new lsp4j.MarkupContent
      markup.setKind("markdown")
      markup.setValue(content.trim)
      markup
    }
  }

  private def hoverContent(typeInfo: Option[String],
                           comments: List[ParsedComment]
                          )(implicit ctx: Context): lsp4j.MarkupContent = {
    val buf = new StringBuilder
    typeInfo.foreach { info =>
      buf.append(s"""```scala
                    |$info
                    |```
                    |""".stripMargin)
    }
    comments.foreach { comment =>
      buf.append(comment.renderAsMarkdown)
    }

    markupContent(buf.toString)
  }

  /** Create an lsp4j.SymbolInfo from a Symbol and a SourcePosition */
  def symbolInfo(sym: Symbol, pos: SourcePosition)(implicit ctx: Context): Option[lsp4j.SymbolInformation] = {
    def symbolKind(sym: Symbol)(implicit ctx: Context): lsp4j.SymbolKind = {
      import lsp4j.{SymbolKind => SK}

      if (sym.is(Package))
        SK.Package
      else if (sym.isConstructor)
        SK.Constructor
      else if (sym.is(Module))
        SK.Module
      else if (sym.isAllOf(EnumCase) || sym.isAllOf(EnumValue))
        SK.EnumMember
      else if (sym.is(Enum))
        SK.Enum
      else if (sym.is(Trait))
        SK.Interface
      else if (sym.isClass)
        SK.Class
      else if (sym.is(Mutable))
        SK.Variable
      else if (sym.is(Method))
        SK.Method
      else if (sym.is(TypeParam) || sym.isAbstractOrAliasType)
        SK.TypeParameter
      else
        SK.Field
    }
    def containerName(sym: Symbol): String = {
      val owner = if (sym.owner.exists && sym.owner.isPackageObject) sym.owner.owner else sym.owner
      if (owner.exists && !owner.isEmptyPackage) {
          owner.name.stripModuleClassSuffix.show
      } else
        null
    }

    val name = sym.name.stripModuleClassSuffix.show

    location(pos).map(l => new lsp4j.SymbolInformation(name, symbolKind(sym), l, containerName(sym)))
  }

  /** Convert `signature` to a `SignatureInformation` */
  def signatureToSignatureInformation(signature: Signatures.Signature): lsp4j.SignatureInformation = {
    val tparams = signature.tparams.map(Signatures.Param("", _))
    val paramInfoss =
      (tparams ::: signature.paramss.flatten).map(paramToParameterInformation)
    val paramLists =
      if signature.paramss.forall(_.isEmpty) && tparams.nonEmpty then
        ""
      else
        signature.paramss
          .map { paramList =>
            val labels = paramList.map(_.show)
            val prefix = if paramList.exists(_.isImplicit) then "using " else ""
            labels.mkString(prefix, ", ", "")
          }
          .mkString("(", ")(", ")")
    val tparamsLabel = if (signature.tparams.isEmpty) "" else signature.tparams.mkString("[", ", ", "]")
    val returnTypeLabel = signature.returnType.map(t => s": $t").getOrElse("")
    val label = s"${signature.name}$tparamsLabel$paramLists$returnTypeLabel"
    val documentation = signature.doc.map(DottyLanguageServer.markupContent)
    val sig = new lsp4j.SignatureInformation(label)
    sig.setParameters(paramInfoss.asJava)
    documentation.foreach(sig.setDocumentation(_))
    sig
  }

  /** Convert `param` to `ParameterInformation` */
  private def paramToParameterInformation(param: Signatures.Param): lsp4j.ParameterInformation = {
    val documentation = param.doc.map(DottyLanguageServer.markupContent)
    val info = new lsp4j.ParameterInformation(param.show)
    documentation.foreach(info.setDocumentation(_))
    info
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy