From 22f0c44813658293ad1cf8a818ed0448c883a43a Mon Sep 17 00:00:00 2001 From: kasiaMarek Date: Tue, 17 Dec 2024 16:59:24 +0100 Subject: [PATCH 1/2] fix: scala 2 additional checks --- .../internal/pc/InferredTypeProvider.scala | 6 +++-- .../internal/pc/SignatureHelpProvider.scala | 25 +++++++++++-------- 2 files changed, 18 insertions(+), 13 deletions(-) diff --git a/mtags/src/main/scala-2/scala/meta/internal/pc/InferredTypeProvider.scala b/mtags/src/main/scala-2/scala/meta/internal/pc/InferredTypeProvider.scala index ecbd687fcfb..15a94337fc0 100644 --- a/mtags/src/main/scala-2/scala/meta/internal/pc/InferredTypeProvider.scala +++ b/mtags/src/main/scala-2/scala/meta/internal/pc/InferredTypeProvider.scala @@ -109,7 +109,8 @@ final class InferredTypeProvider( * turns into * `val a: Int = 1` or `var b: Int = 2` */ - case vl @ ValDef(_, name, tpt, rhs) if !vl.symbol.isParameter => + case vl @ ValDef(_, name, tpt, rhs) + if !vl.symbol.isParameter && tpt.pos.isDefined => val nameEnd = findNameEnd(tpt.pos.start, name) val nameEndPos = tpt.pos.withEnd(nameEnd).withStart(nameEnd).toLsp adjustOpt.foreach(adjust => nameEndPos.setEnd(adjust.adjustedEndPos)) @@ -124,7 +125,8 @@ final class InferredTypeProvider( * turns into * `.map((a: Int) => a + a)` */ - case vl @ ValDef(_, name, tpt, _) if vl.symbol.isParameter => + case vl @ ValDef(_, name, tpt, _) + if vl.symbol.isParameter && tpt.pos.isDefined => val nameEnd = findNameEnd(vl.pos.start, name) val namePos = tpt.pos.withEnd(nameEnd).withStart(nameEnd).toLsp diff --git a/mtags/src/main/scala-2/scala/meta/internal/pc/SignatureHelpProvider.scala b/mtags/src/main/scala-2/scala/meta/internal/pc/SignatureHelpProvider.scala index 8f2df9c7667..d2e020b99a7 100644 --- a/mtags/src/main/scala-2/scala/meta/internal/pc/SignatureHelpProvider.scala +++ b/mtags/src/main/scala-2/scala/meta/internal/pc/SignatureHelpProvider.scala @@ -478,18 +478,21 @@ class SignatureHelpProvider(val compiler: MetalsGlobal) { if (activeSignature == null) { activeSignature = 0 } - val mainSignature = infos(activeSignature) - val deduplicated = infos - .filter { sig => - sig != mainSignature && sig.getLabel() != mainSignature.getLabel() - } - .distinctBy(_.getLabel()) + if (infos.isEmpty) new SignatureHelp() + else { + val mainSignature = infos(activeSignature) + val deduplicated = infos + .filter { sig => + sig != mainSignature && sig.getLabel() != mainSignature.getLabel() + } + .distinctBy(_.getLabel()) - new SignatureHelp( - (mainSignature :: deduplicated).asJava, - 0, - activeParameter - ) + new SignatureHelp( + (mainSignature :: deduplicated).asJava, + 0, + activeParameter + ) + } } def mparamss( From 61144f712d5767bc3048ed1348ab94dacf428bdd Mon Sep 17 00:00:00 2001 From: kasiaMarek Date: Wed, 18 Dec 2024 15:39:02 +0100 Subject: [PATCH 2/2] catch `NullPointerException` in `typedTreeAt` in signature help --- .../internal/pc/SignatureHelpProvider.scala | 22 +++++++++++++------ 1 file changed, 15 insertions(+), 7 deletions(-) diff --git a/mtags/src/main/scala-2/scala/meta/internal/pc/SignatureHelpProvider.scala b/mtags/src/main/scala-2/scala/meta/internal/pc/SignatureHelpProvider.scala index d2e020b99a7..a63dc555de2 100644 --- a/mtags/src/main/scala-2/scala/meta/internal/pc/SignatureHelpProvider.scala +++ b/mtags/src/main/scala-2/scala/meta/internal/pc/SignatureHelpProvider.scala @@ -20,15 +20,23 @@ class SignatureHelpProvider(val compiler: MetalsGlobal) { cursor = cursor(params.offset(), params.text()) ) val pos = unit.position(params.offset()) - typedTreeAt(pos) - val enclosingApply = new EnclosingApply(pos).find(unit.body) - val typedEnclosing = typedTreeAt(enclosingApply.pos) - new MethodCallTraverser(unit, pos) - .fromTree(typedEnclosing) - .map(toSignatureHelp) - .getOrElse(new SignatureHelp()) + (for { + _ <- safeTypedTreeAt(pos) + enclosingApply = new EnclosingApply(pos).find(unit.body) + typedEnclosing <- safeTypedTreeAt(enclosingApply.pos) + encolsingCall <- new MethodCallTraverser(unit, pos).fromTree( + typedEnclosing + ) + } yield toSignatureHelp(encolsingCall)) getOrElse new SignatureHelp() } + private def safeTypedTreeAt(pos: Position): Option[Tree] = + try { + Some(typedTreeAt(pos)) + } catch { + case _: NullPointerException => None + } + class EnclosingApply(pos: Position) extends Traverser { var last: Tree = EmptyTree def find(tree: Tree): Tree = {