/* * Copyright 2014-2023 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license. */ package translators import org.jetbrains.dokka.DokkaConfiguration import org.jetbrains.dokka.DokkaConfiguration.Visibility import org.jetbrains.dokka.base.testApi.testRunner.BaseAbstractTest import org.jetbrains.dokka.links.DRI import org.jetbrains.dokka.links.PointingToDeclaration import org.jetbrains.dokka.model.* import org.jetbrains.dokka.model.doc.* import kotlin.test.* import utils.JavaCode @JavaCode class DefaultPsiToDocumentableTranslatorTest : BaseAbstractTest() { val configuration = dokkaConfiguration { sourceSets { sourceSet { sourceRoots = listOf("src/main/java") } } } @Test fun `method overriding two documented classes picks closest class documentation`() { testInline( """ |/src/main/java/sample/BaseClass1.java |package sample; |public class BaseClass1 { | /** B1 */ | public void x() { } |} | |/src/main/java/sample/BaseClass2.java |package sample; |public class BaseClass2 extends BaseClass1 { | /** B2 */ | public void x() { } |} | |/src/main/java/sample/X.java |package sample; |public class X extends BaseClass2 { | public void x() { } |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val documentationOfFunctionX = module.documentationOf("X", "x") assertTrue( "B2" in documentationOfFunctionX, "Expected nearest super method documentation to be parsed as documentation. " + "Documentation: $documentationOfFunctionX" ) } } } @Test fun `method overriding class and interface picks class documentation`() { testInline( """ |/src/main/java/sample/BaseClass1.java |package sample; |public class BaseClass1 { | /** B1 */ | public void x() { } |} | |/src/main/java/sample/Interface1.java |package sample; |public interface Interface1 { | /** I1 */ | public void x() {} |} | |/src/main/java/sample/X.java |package sample; |public class X extends BaseClass1 implements Interface1 { | public void x() { } |} """.trimMargin(), configuration ) { documentablesMergingStage = { module -> val documentationOfFunctionX = module.documentationOf("X", "x") assertTrue( "B1" in documentationOfFunctionX, "Expected documentation of superclass being prioritized over interface " + "Documentation: $documentationOfFunctionX" ) } } } @Test fun `method overriding two classes picks closest documented class documentation`() { testInline( """ |/src/main/java/sample/BaseClass1.java |package sample; |public class BaseClass1 { | /** B1 */ | public void x() { } |} | |/src/main/java/sample/BaseClass2.java |package sample; |public class BaseClass2 extends BaseClass1 { | public void x() {} |} | |/src/main/java/sample/X.java |package sample; |public class X extends BaseClass2 { | public void x() { } |} """.trimMargin(), configuration ) { documentablesMergingStage = { module -> val documentationOfFunctionX = module.documentationOf("X", "x") assertTrue( "B1" in documentationOfFunctionX, "Expected Documentation \"B1\", found: \"$documentationOfFunctionX\"" ) } } } @Test fun `java package-info package description`() { testInline( """ |/src/main/java/sample/BaseClass1.java |package sample; |public class BaseClass1 { | /** B1 */ | void x() { } |} | |/src/main/java/sample/BaseClass2.java |package sample; |public class BaseClass2 extends BaseClass1 { | void x() {} |} | |/src/main/java/sample/X.java |package sample; |public class X extends BaseClass2 { | void x() { } |} | |/src/main/java/sample/package-info.java |/** | * Here comes description from package-info | */ |package sample; """.trimMargin(), configuration ) { documentablesMergingStage = { module -> val documentationOfPackage = module.packages.single().documentation.values.single().children.single() .firstMemberOfType().body assertEquals( "Here comes description from package-info", documentationOfPackage ) } } } @Test fun `java package-info package annotations`() { testInline( """ |/src/main/java/sample/PackageAnnotation.java |package sample; |@java.lang.annotation.Target(java.lang.annotation.ElementType.PACKAGE) |public @interface PackageAnnotation { |} | |/src/main/java/sample/package-info.java |@PackageAnnotation |package sample; """.trimMargin(), configuration ) { documentablesMergingStage = { module -> assertEquals( Annotations.Annotation(DRI("sample", "PackageAnnotation"), emptyMap()), module.packages.single().extra[Annotations]?.directAnnotations?.values?.single()?.single() ) } } } @Test fun `should add default value to constant properties`() { testInline( """ |/src/main/java/test/JavaConstants.java |package test; | |public class JavaConstants { | public static final byte BYTE = 1; | public static final short SHORT = 2; | public static final int INT = 3; | public static final long LONG = 4L; | public static final float FLOAT = 5.0f; | public static final double DOUBLE = 6.0d; | public static final String STRING = "Seven"; | public static final char CHAR = 'E'; | public static final boolean BOOLEAN = true; |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.packages.single().classlikes.single { it.name == "JavaConstants" } val constants = testedClass.properties assertEquals(9, constants.size) val constantsByName = constants.associateBy { it.name } fun getConstantExpression(name: String): Expression? { return constantsByName.getValue(name).extra[DefaultValue]?.expression?.values?.first() } assertEquals(IntegerConstant(1), getConstantExpression("BYTE")) assertEquals(IntegerConstant(2), getConstantExpression("SHORT")) assertEquals(IntegerConstant(3), getConstantExpression("INT")) assertEquals(IntegerConstant(4), getConstantExpression("LONG")) assertEquals(FloatConstant(5.0f), getConstantExpression("FLOAT")) assertEquals(DoubleConstant(6.0), getConstantExpression("DOUBLE")) assertEquals(StringConstant("Seven"), getConstantExpression("STRING")) assertEquals(StringConstant("E"), getConstantExpression("CHAR")) assertEquals(BooleanConstant(true), getConstantExpression("BOOLEAN")) } } } @Test fun `should resolve static imports used as annotation param values as literal values`() { testInline( """ |/src/main/java/test/JavaClassUsingAnnotation.java |package test; | |import static test.JavaConstants.STRING; |import static test.JavaConstants.INTEGER; |import static test.JavaConstants.LONG; |import static test.JavaConstants.BOOLEAN; |import static test.JavaConstants.DOUBLE; |import static test.JavaConstants.FLOAT; |import static test.JavaConstants.BYTE; |import static test.JavaConstants.SHORT; |import static test.JavaConstants.CHAR; | |@JavaAnnotation( | byteValue = BYTE, shortValue = SHORT, intValue = INTEGER, longValue = LONG, booleanValue = BOOLEAN, | doubleValue = DOUBLE, floatValue = FLOAT, stringValue = STRING, charValue = CHAR |) |public class JavaClassUsingAnnotation { |} | |/src/main/java/test/JavaAnnotation.java |package test; |@Documented |public @interface JavaAnnotation { | byte byteValue(); | short shortValue(); | int intValue(); | long longValue(); | boolean booleanValue(); | double doubleValue(); | float floatValue(); | String stringValue(); | char charValue(); |} | |/src/main/java/test/JavaConstants.java |package test; |public class JavaConstants { | public static final byte BYTE = 3; | public static final short SHORT = 4; | public static final int INTEGER = 5; | public static final long LONG = 6L; | public static final boolean BOOLEAN = true; | public static final double DOUBLE = 7.0d; | public static final float FLOAT = 8.0f; | public static final String STRING = "STRING_CONSTANT_VALUE"; | public static final char CHAR = 'c'; |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.packages.single().classlikes.single { it.name == "JavaClassUsingAnnotation" } val annotation = (testedClass as DClass).extra[Annotations]?.directAnnotations?.values?.single()?.single() assertNotNull(annotation) assertEquals("JavaAnnotation", annotation.dri.classNames) assertEquals(IntValue(3), annotation.params["byteValue"]) assertEquals(IntValue(4), annotation.params["shortValue"]) assertEquals(IntValue(5), annotation.params["intValue"]) assertEquals(LongValue(6), annotation.params["longValue"]) assertEquals(BooleanValue(true), annotation.params["booleanValue"]) assertEquals(DoubleValue(7.0), annotation.params["doubleValue"]) assertEquals(FloatValue(8.0f), annotation.params["floatValue"]) assertEquals(StringValue("STRING_CONSTANT_VALUE"), annotation.params["stringValue"]) assertEquals(StringValue("c"), annotation.params["charValue"]) } } } // TODO [beresnev] fix // class OnlyPsiPlugin : DokkaPlugin() { // private val kotlinAnalysisPlugin by lazy { plugin() } // // @Suppress("unused") // val psiOverrideDescriptorTranslator by extending { // (plugin().psiToDocumentableTranslator // override kotlinAnalysisPlugin.descriptorToDocumentableTranslator) // } // // @OptIn(DokkaPluginApiPreview::class) // override fun pluginApiPreviewAcknowledgement(): PluginApiPreviewAcknowledgement = // PluginApiPreviewAcknowledgement // } // // // for Kotlin classes from DefaultPsiToDocumentableTranslator // @Test // fun `should resolve ultralight class`() { // val configurationWithNoJVM = dokkaConfiguration { // sourceSets { // sourceSet { // sourceRoots = listOf("src/main/java") // } // } // } // // testInline( // """ // |/src/main/java/example/Test.kt // |package example // | // |open class KotlinSubClass { // | fun kotlinSubclassFunction(bar: String): String { // | return "KotlinSubClass" // | } // |} // | // |/src/main/java/example/JavaLeafClass.java // |package example; // | // |public class JavaLeafClass extends KotlinSubClass { // | public String javaLeafClassFunction(String baz) { // | return "JavaLeafClass"; // | } // |} // """.trimMargin(), // configurationWithNoJVM, // pluginOverrides = listOf(OnlyPsiPlugin()) // suppress a descriptor translator because of psi and descriptor translators work in parallel // ) { // documentablesMergingStage = { module -> // val kotlinSubclassFunction = // module.packages.single().classlikes.find { it.name == "JavaLeafClass" }?.functions?.find { it.name == "kotlinSubclassFunction" } // .assertNotNull("kotlinSubclassFunction ") // // assertEquals( // "String", // (kotlinSubclassFunction.type as? TypeConstructor)?.dri?.classNames // ) // assertEquals( // "String", // (kotlinSubclassFunction.parameters.firstOrNull()?.type as? TypeConstructor)?.dri?.classNames // ) // } // } // } @Test fun `should preserve regular functions that are named like getters, but are not getters`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { | private int a = 1; | public String getA() { return "s"; } // wrong return type | public int getA(String param) { return 123; } // shouldn't have params |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testClass = module.packages.single().classlikes.single { it.name == "A" } val getterLookalikes = testClass.functions.filter { it.name == "getA" } assertEquals(2, getterLookalikes.size, "Not all expected regular functions found, wrongly categorized as getters?") } } } @Test fun `should ignore additional non-accessor setters`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { | private int a = 1; | | public int getA() { return a; } | | public void setA(long a) { } | public void setA(Number a) {} | | // the qualifying setter is intentionally in the middle | // to rule out the order making a difference | public void setA(int a) { } | | public void setA(String a) {} | public void setA() {} | |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testClass = module.packages.single().classlikes.single { it.name == "A" } val property = testClass.properties.single { it.name == "a" } assertNotNull(property.getter) val setter = property.setter assertNotNull(setter) assertEquals(1, setter.parameters.size) assertEquals(PrimitiveJavaType("int"), setter.parameters[0].type) val regularSetterFunctions = testClass.functions.filter { it.name == "setA" } assertEquals(4, regularSetterFunctions.size) } } } @Test fun `should not qualify methods with subtype parameters as type accessors`() { testInline( """ |/src/main/java/test/Shape.java |package test; |public class Shape { } | |/src/main/java/test/Triangle.java |package test; |public class Triangle extends Shape { } | |/src/main/java/test/Square.java |package test; |public class Square extends Shape { } | |/src/main/java/test/Test.java |package test; |public class Test { | private Shape foo = 1; | | public Shape getFoo() { return new Square(); } | | public void setFoo(Square foo) { } | public void setFoo(Triangle foo) { } |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testClass = module.packages.single().classlikes.single { it.name == "Test" } val field = testClass.properties.singleOrNull { it.name == "foo" } assertNotNull(field) { "Expected the foo property to exist because the field is private with a public getter" } assertNull(field.setter) val setterMethodsWithSubtypeParams = testClass.functions.filter { it.name == "setFoo" } assertEquals( 2, setterMethodsWithSubtypeParams.size, "Expected the setter methods to not qualify as accessors because of subtype parameters" ) } } } @Test fun `should preserve private fields without getters even if they have qualifying setters`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { | private int a = 1; | | public void setA(int a) { } |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val tetClass = module.packages.single().classlikes.single { it.name == "A" } val property = tetClass.properties.firstOrNull { it.name == "a" } assertNull(property, "Expected the property to stay private because there are no getters") val regularSetterFunction = tetClass.functions.firstOrNull { it.name == "setA" } assertNotNull(regularSetterFunction) { "The qualifying setter function should stay a regular function because the field is inaccessible" } } } } @Test fun `should not mark a multi-param setter overload as an accessor`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { | private int field = 1; | | public void setField(int a, int b) { } | public int getField() { return a; } |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testClass = module.packages.single().classlikes.single { it.name == "A" } as DClass val property = testClass.properties.single { it.name == "field" } assertEquals("getField", property.getter?.name) assertNull(property.setter) // the setField function should not qualify to be an accessor due to the second param assertEquals(1, testClass.functions.size) assertEquals("setField", testClass.functions[0].name) } } } @Test fun `should not associate accessors with field because field is public api`() { val configuration = dokkaConfiguration { sourceSets { sourceSet { sourceRoots = listOf("src/") documentedVisibilities = setOf( DokkaConfiguration.Visibility.PUBLIC, DokkaConfiguration.Visibility.PROTECTED ) } } } testInline( """ |/src/test/A.java |package test; |public class A { | protected int a = 1; | public int getA() { return a; } | public void setA(int a) { this.a = a; } |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.packages.single().classlikes.single { it.name == "A" } val property = testedClass.properties.single { it.name == "a" } assertEquals(JavaVisibility.Protected, property.visibility.values.single()) assertNull(property.getter) assertNull(property.setter) assertEquals(2, testedClass.functions.size) assertEquals("getA", testedClass.functions[0].name) assertEquals("setA", testedClass.functions[1].name) } } } @Test fun `should add IsVar extra for field with getter and setter`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { | private int a = 1; | public int getA() { return a; } | public void setA(int a) { this.a = a; } |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.packages.single().classlikes.single { it.name == "A" } val property = testedClass.properties.single { it.name == "a" } assertNotNull(property.extra[IsVar]) } } } @Test fun `should not add IsVar extra if field does not have a setter`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { | private int a = 1; | public int getA() { return a; } |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.packages.single().classlikes.single { it.name == "A" } val property = testedClass.properties.single { it.name == "a" } assertNull(property.extra[IsVar]) } } } @Test fun `should add IsVar for non-final java field without any accessors`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { | public int a = 1; |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.packages.single().classlikes.single { it.name == "A" } val property = testedClass.properties.single { it.name == "a" } assertNotNull(property.extra[IsVar]) } } } @Test fun `should not add IsVar for final java field`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { | public final int a = 2; |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.packages.single().classlikes.single { it.name == "A" } val publicFinal = testedClass.properties.single { it.name == "a" } assertNull(publicFinal.extra[IsVar]) } } } @Test // see https://github.com/Kotlin/dokka/issues/2646 fun `should resolve PsiImmediateClassType as class reference`() { testInline( """ |/src/main/java/test/JavaEnum.java |package test; |public enum JavaEnum { | FOO, BAR |} | |/src/main/java/test/ContainingEnumType.java |package test; |public class ContainingEnumType { | | public JavaEnum returningEnumType() { | return null; | } | | public JavaEnum[] returningEnumTypeArray() { | return null; | } | | public void acceptingEnumType(JavaEnum javaEnum) {} |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val expectedType = GenericTypeConstructor( dri = DRI(packageName = "test", classNames = "JavaEnum", target = PointingToDeclaration), projections = emptyList() ) val expectedArrayType = GenericTypeConstructor( dri = DRI("kotlin", "Array", target = PointingToDeclaration), projections = listOf(expectedType) ) val classWithEnumUsage = module.packages.single().classlikes.single { it.name == "ContainingEnumType" } val returningEnum = classWithEnumUsage.functions.single { it.name == "returningEnumType" } assertEquals(expectedType, returningEnum.type) val acceptingEnum = classWithEnumUsage.functions.single { it.name == "acceptingEnumType" } assertEquals(1, acceptingEnum.parameters.size) assertEquals(expectedType, acceptingEnum.parameters[0].type) val returningArray = classWithEnumUsage.functions.single { it.name == "returningEnumTypeArray" } assertEquals(expectedArrayType, returningArray.type) } } } @Test fun `should have documentation for synthetic Enum values functions`() { testInline( """ |/src/main/java/test/JavaEnum.java |package test | |public enum JavaEnum { | FOO, BAR; |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val kotlinEnum = module.packages.find { it.name == "test" } ?.classlikes ?.single { it.name == "JavaEnum" } assertNotNull(kotlinEnum) val valuesFunction = kotlinEnum.functions.single { it.name == "values" } val expectedDocumentation = DocumentationNode(listOf( Description( CustomDocTag( children = listOf( P(listOf( Text( "Returns an array containing the constants of this enum type, " + "in the order they're declared. This method may be used to " + "iterate over the constants." ), )) ), name = "MARKDOWN_FILE" ) ), Return( CustomDocTag( children = listOf( P(listOf( Text("an array containing the constants of this enum type, in the order they're declared") )) ), name = "MARKDOWN_FILE" ) ) )) assertEquals(expectedDocumentation, valuesFunction.documentation.values.single()) val expectedValuesType = GenericTypeConstructor( dri = DRI( packageName = "kotlin", classNames = "Array" ), projections = listOf( GenericTypeConstructor( dri = DRI( packageName = "test", classNames = "JavaEnum" ), projections = emptyList() ) ) ) assertEquals(expectedValuesType, valuesFunction.type) } } } @Test fun `should have documentation for synthetic Enum valueOf functions`() { testInline( """ |/src/main/java/test/JavaEnum.java |package test | |public enum JavaEnum { | FOO, BAR; |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val javaEnum = module.packages.find { it.name == "test" } ?.classlikes ?.single { it.name == "JavaEnum" } assertNotNull(javaEnum) val valueOfFunction = javaEnum.functions.single { it.name == "valueOf" } val expectedDocumentation = DocumentationNode(listOf( Description( CustomDocTag( children = listOf( P(listOf( Text( "Returns the enum constant of this type with the " + "specified name. The string must match exactly an identifier used " + "to declare an enum constant in this type. (Extraneous whitespace " + "characters are not permitted.)" ) )) ), name = "MARKDOWN_FILE" ) ), Return( root = CustomDocTag( children = listOf( P(listOf( Text("the enum constant with the specified name") )) ), name = "MARKDOWN_FILE" ) ), Throws( name = "java.lang.IllegalArgumentException", exceptionAddress = DRI( packageName = "java.lang", classNames = "IllegalArgumentException", target = PointingToDeclaration ), root = CustomDocTag( children = listOf( P(listOf( Text("if this enum type has no constant with the specified name") )) ), name = "MARKDOWN_FILE" ) ), )) assertEquals(expectedDocumentation, valueOfFunction.documentation.values.single()) val expectedValueOfType = GenericTypeConstructor( dri = DRI( packageName = "test", classNames = "JavaEnum" ), projections = emptyList() ) assertEquals(expectedValueOfType, valueOfFunction.type) val valueOfParamDRI = (valueOfFunction.parameters.single().type as GenericTypeConstructor).dri assertEquals(DRI(packageName = "java.lang", classNames = "String"), valueOfParamDRI) } } } @Test fun `should have public default constructor in public class`() { testInline( """ |/src/main/java/test/A.java |package test; |public class A { |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.findClasslike(packageName = "test", "A") as DClass assertEquals(1, testedClass.constructors.size, "Expect 1 default constructor") assertTrue( testedClass.constructors.first().parameters.isEmpty(), "Expect default constructor doesn't have params" ) assertEquals(JavaVisibility.Public, testedClass.constructors.first().visibility()) } } } @Test fun `should have package-private default constructor in package-private class`() { val configuration = dokkaConfiguration { sourceSets { sourceSet { sourceRoots = listOf("src/main/java") documentedVisibilities = setOf(Visibility.PUBLIC, Visibility.PACKAGE) } } } testInline( """ |/src/main/java/test/A.java |package test; |class A { |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.findClasslike(packageName = "test", "A") as DClass assertEquals(1, testedClass.constructors.size, "Expect 1 default constructor") assertEquals(JavaVisibility.Default, testedClass.constructors.first().visibility()) } } } @Test fun `should have private default constructor in private nested class`() { val configuration = dokkaConfiguration { sourceSets { sourceSet { sourceRoots = listOf("src/main/java") documentedVisibilities = setOf(Visibility.PUBLIC, Visibility.PRIVATE) } } } testInline( """ |/src/main/java/test/A.java |package test; |public class A { | private static class PrivateNested{} |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val parentClass = module.findClasslike(packageName = "test", "A") as DClass val testedClass = parentClass.classlikes.single { it.name == "PrivateNested" } as DClass assertEquals(1, testedClass.constructors.size, "Expect 1 default constructor") assertEquals(JavaVisibility.Private, testedClass.constructors.first().visibility()) } } } @Test fun `should not have a default public constructor because have explicit private`() { val configuration = dokkaConfiguration { sourceSets { sourceSet { sourceRoots = listOf("src/main/java") documentedVisibilities = setOf(Visibility.PUBLIC, Visibility.PRIVATE) } } } testInline( """ |/src/main/java/test/A.java |package test; |public class A { | private A(){} |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.findClasslike(packageName = "test", "A") as DClass assertEquals(1, testedClass.constructors.size, "Expect 1 declared constructor") assertEquals(JavaVisibility.Private, testedClass.constructors.first().visibility()) } } } @Test fun `default constructor should get the package name`() { testInline( """ |/src/main/java/org/test/A.java |package org.test; |public class A { |} """.trimIndent(), configuration ) { documentablesMergingStage = { module -> val testedClass = module.findClasslike(packageName = "org.test", "A") as DClass assertEquals(1, testedClass.constructors.size, "Expect 1 default constructor") val constructorDRI = testedClass.constructors.first().dri assertEquals("org.test", constructorDRI.packageName) assertEquals("A", constructorDRI.classNames) } } } } private fun DFunction.visibility() = visibility.values.first()