Skip to content

Commit 56463fd

Browse files
authored
markus/koltinBugfixes (#5200)
* Fix handling of lambda functions wrapped by constructs like labels. This resulted in a NoSuchElementException because the lambda expression could not be associated with the correct call argument. * Handle case where a type constructor has no declaration descriptor. This case is unexpected. We now avoid the MatchErrorException return None and log in order to not crash and burn and to understand the situation better. * Fix broken CPG in case variable declaration without initialization.
1 parent cc0785b commit 56463fd

File tree

5 files changed

+104
-12
lines changed

5 files changed

+104
-12
lines changed

joern-cli/frontends/kotlin2cpg/src/main/scala/io/joern/kotlin2cpg/ast/AstForDeclarationsCreator.scala

+13-9
Original file line numberDiff line numberDiff line change
@@ -713,15 +713,19 @@ trait AstForDeclarationsCreator(implicit withSchemaValidation: ValidationMode) {
713713
scope.addToScope(expr.getName, node)
714714
val localAst = Ast(node)
715715

716-
val rhsAsts = astsForExpression(expr.getDelegateExpressionOrInitializer, Some(2))
717-
val identifier = identifierNode(elem, elem.getText, elem.getText, typeFullName)
718-
val identifierAst = astWithRefEdgeMaybe(identifier.name, identifier)
719-
val assignmentNode =
720-
NodeBuilders.newOperatorCallNode(Operators.assignment, expr.getText, None, line(expr), column(expr))
721-
val call =
722-
callAst(assignmentNode, List(identifierAst) ++ rhsAsts)
723-
.withChildren(annotations.map(astForAnnotationEntry))
724-
Seq(localAst, call)
716+
if (expr.getDelegateExpressionOrInitializer != null) {
717+
val rhsAsts = astsForExpression(expr.getDelegateExpressionOrInitializer, Some(2))
718+
val identifier = identifierNode(elem, elem.getText, elem.getText, typeFullName)
719+
val identifierAst = astWithRefEdgeMaybe(identifier.name, identifier)
720+
val assignmentNode =
721+
NodeBuilders.newOperatorCallNode(Operators.assignment, expr.getText, None, line(expr), column(expr))
722+
val call =
723+
callAst(assignmentNode, List(identifierAst) ++ rhsAsts)
724+
.withChildren(annotations.map(astForAnnotationEntry))
725+
Seq(localAst, call)
726+
} else {
727+
Seq(localAst)
728+
}
725729
}
726730
}
727731

joern-cli/frontends/kotlin2cpg/src/main/scala/io/joern/kotlin2cpg/ast/AstForFunctionsCreator.scala

+35-2
Original file line numberDiff line numberDiff line change
@@ -556,9 +556,19 @@ trait AstForFunctionsCreator(implicit withSchemaValidation: ValidationMode) {
556556
// Lambda is wrapped e.g. `SomeInterface { obj -> obj }`
557557
samConstructorDesc.getBaseDescriptorForSynthetic
558558
case _ =>
559-
// Lambda/anan function is directly used as call argument e.g. `someCall(obj -> obj)`
559+
// Lambda/anon function is directly used as call argument e.g. `someCall(obj -> obj)`
560+
val directCallArgumentForLookup =
561+
expr match {
562+
case _: KtNamedFunction =>
563+
// This is the anonymous function case.
564+
// So far it does not seem like those could be wrapped so they are always the direct argument.
565+
expr
566+
case _ =>
567+
// The lambda function case.
568+
getDirectLambdaArgument(expr).get
569+
}
560570
callAtom.getArgumentMappingByOriginal.asScala.collectFirst {
561-
case (paramDesc, resolvedArgument) if isExprIncluded(resolvedArgument, expr) =>
571+
case (paramDesc, resolvedArgument) if isExprIncluded(resolvedArgument, directCallArgumentForLookup) =>
562572
paramDesc.getType.getConstructor.getDeclarationDescriptor.asInstanceOf[ClassDescriptor]
563573
}.get
564574
}
@@ -571,6 +581,28 @@ trait AstForFunctionsCreator(implicit withSchemaValidation: ValidationMode) {
571581
}
572582
}
573583

584+
private def getDirectLambdaArgument(element: KtElement): Option[KtExpression] = {
585+
var context: PsiElement = element
586+
var parentContext: PsiElement = null
587+
588+
// KtCallExpression wrap their arguments in KtValueArgument which is why we look for those.
589+
// KtBinaryExpressions do not do such a wrapping.
590+
while ({
591+
parentContext = context.getContext
592+
parentContext != null &&
593+
!parentContext.isInstanceOf[KtValueArgument] &&
594+
!parentContext.isInstanceOf[KtBinaryExpression]
595+
}) {
596+
context = parentContext
597+
}
598+
599+
if (parentContext != null) {
600+
Some(context.asInstanceOf[KtExpression])
601+
} else {
602+
None
603+
}
604+
}
605+
574606
private def getSurroundingCallTarget(element: KtElement): Option[KtExpression] = {
575607
var context: PsiElement = element.getContext
576608
while (
@@ -591,6 +623,7 @@ trait AstForFunctionsCreator(implicit withSchemaValidation: ValidationMode) {
591623
}
592624

593625
private def isExprIncluded(resolvedArgument: ResolvedCallArgument, expr: KtExpression): Boolean = {
626+
// getArguments returns multiple arguments in case of varargs
594627
resolvedArgument.getArguments.asScala.exists {
595628
case psi: PSIFunctionKotlinCallArgument =>
596629
psi.getExpression == expr

joern-cli/frontends/kotlin2cpg/src/main/scala/io/joern/kotlin2cpg/types/NameRenderer.scala

+11-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
package io.joern.kotlin2cpg.types
22

3-
import io.joern.kotlin2cpg.types.NameRenderer.BuiltinTypeTranslationTable
3+
import io.joern.kotlin2cpg.types.NameRenderer.{BuiltinTypeTranslationTable, logger}
44
import io.joern.x2cpg.Defines
55
import org.jetbrains.kotlin.builtins.jvm.JavaToKotlinClassMap
66
import org.jetbrains.kotlin.descriptors.impl.TypeAliasConstructorDescriptor
@@ -16,11 +16,14 @@ import org.jetbrains.kotlin.descriptors.{
1616
import org.jetbrains.kotlin.name.FqNameUnsafe
1717
import org.jetbrains.kotlin.types.KotlinType
1818
import org.jetbrains.kotlin.types.error.ErrorClassDescriptor
19+
import org.slf4j.LoggerFactory
1920

2021
import scala.collection.mutable
2122
import scala.jdk.CollectionConverters.*
2223

2324
object NameRenderer {
25+
private val logger = LoggerFactory.getLogger(getClass)
26+
2427
private val BuiltinTypeTranslationTable = mutable.HashMap(
2528
"kotlin.Unit" -> "void",
2629
"kotlin.Boolean" -> "boolean",
@@ -111,6 +114,13 @@ class NameRenderer {
111114
} else {
112115
Some(upperBoundTypeFns.flatten.mkString("&"))
113116
}
117+
case null =>
118+
// We do not expect this because to my understanding a typ should always have a constructor
119+
// descriptor.
120+
logger.warn(
121+
s"Found type without constructor descriptor. Typ: $typ Constructor class: ${typ.getConstructor.getClass}"
122+
)
123+
None
114124
}
115125

116126
javaFullName

joern-cli/frontends/kotlin2cpg/src/test/scala/io/joern/kotlin2cpg/querying/LambdaTests.scala

+29
Original file line numberDiff line numberDiff line change
@@ -748,4 +748,33 @@ class LambdaTests extends KotlinCode2CpgFixture(withOssDataflow = false, withDef
748748
binding2.bindingTypeDecl shouldBe lambdaTypeDecl
749749
}
750750
}
751+
752+
"CPG for code with lambda wrapped in label" should {
753+
val cpg = code("""
754+
|package mypkg
755+
|fun outer() {
756+
| listOf(1).forEach someLabel@{x: Int -> x}
757+
|}
758+
|""".stripMargin)
759+
760+
"contain correct lambda, bindings and type decl nodes" in {
761+
val List(lambdaMethod) = cpg.method.fullName(".*lambda.*").l
762+
lambdaMethod.fullName shouldBe s"mypkg.outer.${Defines.ClosurePrefix}0:void(int)"
763+
lambdaMethod.signature shouldBe "void(int)"
764+
765+
val List(lambdaTypeDecl) = lambdaMethod.bindingTypeDecl.dedup.l
766+
lambdaTypeDecl.fullName shouldBe s"mypkg.outer.${Defines.ClosurePrefix}0"
767+
lambdaTypeDecl.inheritsFromTypeFullName should contain theSameElementsAs (List("kotlin.Function1"))
768+
769+
val List(binding1, binding2) = lambdaMethod.referencingBinding.l
770+
binding1.name shouldBe "invoke"
771+
binding1.signature shouldBe "void(int)"
772+
binding1.methodFullName shouldBe s"mypkg.outer.${Defines.ClosurePrefix}0:void(int)"
773+
binding1.bindingTypeDecl shouldBe lambdaTypeDecl
774+
binding2.name shouldBe "invoke"
775+
binding2.signature shouldBe "java.lang.Object(java.lang.Object)"
776+
binding2.methodFullName shouldBe s"mypkg.outer.${Defines.ClosurePrefix}0:void(int)"
777+
binding2.bindingTypeDecl shouldBe lambdaTypeDecl
778+
}
779+
}
751780
}

joern-cli/frontends/kotlin2cpg/src/test/scala/io/joern/kotlin2cpg/querying/LocalTests.scala

+16
Original file line numberDiff line numberDiff line change
@@ -25,4 +25,20 @@ class LocalTests extends KotlinCode2CpgFixture(withOssDataflow = false) {
2525
l2.typeFullName shouldBe "int"
2626
}
2727
}
28+
29+
"CPG for local declaration without initialization" should {
30+
val cpg = code("""
31+
|fun main() {
32+
| var x: Int
33+
|}
34+
|""".stripMargin)
35+
36+
"contain LOCAL node for `x`" in {
37+
val List(l1) = cpg.local("x").l
38+
l1.code shouldBe "x"
39+
l1.name shouldBe "x"
40+
l1.typeFullName shouldBe "int"
41+
}
42+
}
43+
2844
}

0 commit comments

Comments
 (0)