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

org.jetbrains.kotlin.load.java.typeEnhancement.predefinedEnhancementInfo.kt Maven / Gradle / Ivy

There is a newer version: 2.0.0
Show newest version
/*
 * Copyright 2010-2016 JetBrains s.r.o.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.jetbrains.kotlin.load.java.typeEnhancement

import org.jetbrains.kotlin.load.kotlin.SignatureBuildingComponents
import org.jetbrains.kotlin.load.kotlin.signatures
import org.jetbrains.kotlin.resolve.jvm.JvmPrimitiveType
import org.jetbrains.kotlin.resolve.jvm.JvmPrimitiveType.BOOLEAN

class TypeEnhancementInfo(val map: Map) {
    constructor(vararg pairs: Pair) : this(mapOf(*pairs))
}

class PredefinedFunctionEnhancementInfo(
    val returnTypeInfo: TypeEnhancementInfo? = null,
    val parametersInfo: List = emptyList()
)

/** Type is always nullable: `T?` */
private val NULLABLE = JavaTypeQualifiers(NullabilityQualifier.NULLABLE, null, isNotNullTypeParameter = false)
/** Nullability depends on substitution, but the type is not platform: `T` */
private val NOT_PLATFORM = JavaTypeQualifiers(NullabilityQualifier.NOT_NULL, null, isNotNullTypeParameter = false)
/** Type is always non-nullable: `T & Any` */
private val NOT_NULLABLE = JavaTypeQualifiers(NullabilityQualifier.NOT_NULL, null, isNotNullTypeParameter = true)

@Suppress("LocalVariableName")
val PREDEFINED_FUNCTION_ENHANCEMENT_INFO_BY_SIGNATURE = signatures {
    val JLObject = javaLang("Object")
    val JFPredicate = javaFunction("Predicate")
    val JFFunction = javaFunction("Function")
    val JFConsumer = javaFunction("Consumer")
    val JFBiFunction = javaFunction("BiFunction")
    val JFBiConsumer = javaFunction("BiConsumer")
    val JFUnaryOperator = javaFunction("UnaryOperator")
    val JUStream = javaUtil("stream/Stream")
    val JUOptional = javaUtil("Optional")

    enhancement {
        forClass(javaUtil("Iterator")) {
            function("forEachRemaining") {
                parameter(JFConsumer, NOT_PLATFORM, NOT_PLATFORM)
            }
        }
        forClass(javaLang("Iterable")) {
            function("spliterator") {
                returns(javaUtil("Spliterator"), NOT_PLATFORM, NOT_PLATFORM)
            }
        }
        forClass(javaUtil("Collection")) {
            function("removeIf") {
                parameter(JFPredicate, NOT_PLATFORM, NOT_PLATFORM)
                returns(BOOLEAN)
            }
            function("stream") {
                returns(JUStream, NOT_PLATFORM, NOT_PLATFORM)
            }
            function("parallelStream") {
                returns(JUStream, NOT_PLATFORM, NOT_PLATFORM)
            }
        }
        forClass(javaUtil("List")) {
            function("replaceAll") {
                parameter(JFUnaryOperator, NOT_PLATFORM, NOT_PLATFORM)
            }
        }
        forClass(javaUtil("Map")) {
            function("forEach") {
                parameter(JFBiConsumer, NOT_PLATFORM, NOT_PLATFORM, NOT_PLATFORM)
            }
            function("putIfAbsent") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JLObject, NOT_PLATFORM)
                returns(JLObject, NULLABLE)
            }
            function("replace") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JLObject, NOT_PLATFORM)
                returns(JLObject, NULLABLE)

            }
            function("replace") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JLObject, NOT_PLATFORM)
                parameter(JLObject, NOT_PLATFORM)
                returns(BOOLEAN)
            }
            function("replaceAll") {
                parameter(JFBiFunction, NOT_PLATFORM, NOT_PLATFORM, NOT_PLATFORM, NOT_PLATFORM)
            }
            function("compute") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JFBiFunction, NOT_PLATFORM, NOT_PLATFORM, NULLABLE, NULLABLE)
                returns(JLObject, NULLABLE)
            }
            // while it is possible to return nullable value from lambda in computeIfAbsent,
            // we deliberately make it just NOT_PLATFORM V in order to have the return type V and not V?
            function("computeIfAbsent") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JFFunction, NOT_PLATFORM, NOT_PLATFORM, NOT_PLATFORM)
                returns(JLObject, NOT_PLATFORM)
            }
            function("computeIfPresent") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JFBiFunction, NOT_PLATFORM, NOT_PLATFORM, NOT_NULLABLE, NULLABLE)
                returns(JLObject, NULLABLE)
            }
            function("merge") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JLObject, NOT_NULLABLE)
                parameter(JFBiFunction, NOT_PLATFORM, NOT_NULLABLE, NOT_NULLABLE, NULLABLE)
                returns(JLObject, NULLABLE)
            }
        }
        forClass(JUOptional) {
            function("empty") {
                returns(JUOptional, NOT_PLATFORM, NOT_NULLABLE)
            }
            function("of") {
                parameter(JLObject, NOT_NULLABLE)
                returns(JUOptional, NOT_PLATFORM, NOT_NULLABLE)
            }
            function("ofNullable") {
                parameter(JLObject, NULLABLE)
                returns(JUOptional, NOT_PLATFORM, NOT_NULLABLE)
            }
            function("get") {
                returns(JLObject, NOT_NULLABLE)
            }
            function("ifPresent") {
                parameter(JFConsumer, NOT_PLATFORM, NOT_NULLABLE)
            }
        }

        forClass(javaLang("ref/Reference")) {
            function("get") {
                returns(JLObject, NULLABLE)
            }
        }

        forClass(JFPredicate) {
            function("test") {
                parameter(JLObject, NOT_PLATFORM)
                returns(BOOLEAN)
            }
        }
        forClass(javaFunction("BiPredicate")) {
            function("test") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JLObject, NOT_PLATFORM)
                returns(BOOLEAN)
            }
        }
        forClass(JFConsumer) {
            function("accept") {
                parameter(JLObject, NOT_PLATFORM)
            }
        }
        forClass(JFBiConsumer) {
            function("accept") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JLObject, NOT_PLATFORM)
            }
        }
        forClass(JFFunction) {
            function("apply") {
                parameter(JLObject, NOT_PLATFORM)
                returns(JLObject, NOT_PLATFORM)
            }
        }
        forClass(JFBiFunction) {
            function("apply") {
                parameter(JLObject, NOT_PLATFORM)
                parameter(JLObject, NOT_PLATFORM)
                returns(JLObject, NOT_PLATFORM)
            }
        }
        forClass(javaFunction("Supplier")) {
            function("get") {
                returns(JLObject, NOT_PLATFORM)
            }
        }
    }
}


private inline fun enhancement(block: SignatureEnhancementBuilder.() -> Unit): Map =
    SignatureEnhancementBuilder().apply(block).build()

private class SignatureEnhancementBuilder {
    private val signatures = mutableMapOf()

    inline fun forClass(internalName: String, block: ClassEnhancementBuilder.() -> Unit) =
        ClassEnhancementBuilder(internalName).block()

    inner class ClassEnhancementBuilder(val className: String) {
        fun function(name: String, block: FunctionEnhancementBuilder.() -> Unit) {
            signatures += FunctionEnhancementBuilder(name).apply(block).build()
        }

        inner class FunctionEnhancementBuilder(val functionName: String) {
            private val parameters = mutableListOf>()
            private var returnType: Pair = "V" to null

            fun parameter(type: String, vararg pairs: Pair) {
                parameters += type to
                        if (pairs.isEmpty()) null else TypeEnhancementInfo(*pairs)
            }

            fun parameter(type: String, vararg qualifiers: JavaTypeQualifiers) {
                parameters += type to
                        if (qualifiers.isEmpty()) null else TypeEnhancementInfo(
                            qualifiers.withIndex().associateBy(
                                { it.index },
                                { it.value })
                        )
            }

            fun returns(type: String, vararg pairs: Pair) {
                returnType = type to TypeEnhancementInfo(*pairs)
            }

            fun returns(type: String, vararg qualifiers: JavaTypeQualifiers) {
                returnType = type to TypeEnhancementInfo(qualifiers.withIndex().associateBy({ it.index }, { it.value }))
            }

            fun returns(type: JvmPrimitiveType) {
                returnType = type.desc to null
            }

            fun build() = with(SignatureBuildingComponents) {
                signature(className, jvmDescriptor(functionName, parameters.map { it.first }, returnType.first)) to
                        PredefinedFunctionEnhancementInfo(returnType.second, parameters.map { it.second })
            }
        }

    }

    fun build(): Map = signatures
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy