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

org.jetbrains.kotlin.js.naming.encodeSignature.kt Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2010-2017 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.js.naming

import org.jetbrains.kotlin.builtins.KotlinBuiltIns
import org.jetbrains.kotlin.builtins.isSuspendFunctionType
import org.jetbrains.kotlin.descriptors.*
import org.jetbrains.kotlin.resolve.DescriptorUtils
import org.jetbrains.kotlin.types.KotlinType
import org.jetbrains.kotlin.types.TypeProjection
import org.jetbrains.kotlin.types.Variance
import org.jetbrains.kotlin.types.getEffectiveVariance

fun encodeSignature(descriptor: CallableDescriptor): String {
    val sig = StringBuilder()

    val typeParameterNames = nameTypeParameters(descriptor)
    val currentParameters = descriptor.typeParameters.filter { !it.isCapturedFromOuterDeclaration }.toSet()
    val usedTypeParameters = currentParameters.toMutableSet()
    val typeParameterNamer = { typeParameter: TypeParameterDescriptor ->
        usedTypeParameters += typeParameter.original
        typeParameterNames[typeParameter.original]
            ?: error("${typeParameter.original} is not found when encode the signature of $descriptor.")
    }

    val contextReceiverParameters = descriptor.contextReceiverParameters
    if (contextReceiverParameters.isNotEmpty()) {
        for (contextReceiverParameter in contextReceiverParameters) {
            sig.encodeForSignature(contextReceiverParameter.type, typeParameterNamer).append(',')
        }
        sig.append('\\')
    }

    val receiverParameter = descriptor.extensionReceiverParameter
    if (receiverParameter != null) {
        sig.encodeForSignature(receiverParameter.type, typeParameterNamer).append('/')
    }

    for (valueParameter in descriptor.valueParameters) {
        if (valueParameter.index > 0) {
            sig.append(",")
        }
        if (valueParameter.varargElementType != null) {
            sig.append("*")
        }
        sig.encodeForSignature(valueParameter.type, typeParameterNamer)
    }

    var first = true
    for (typeParameter in typeParameterNames.keys.asSequence().filter { it in usedTypeParameters }) {
        val upperBounds = typeParameter.upperBounds.filter { !KotlinBuiltIns.isNullableAny(it) }
        if (upperBounds.isEmpty() && typeParameter !in currentParameters) continue

        sig.append(if (first) "|" else ",").append(typeParameterNames[typeParameter])
        first = false
        if (upperBounds.isEmpty()) continue

        sig.append("<:")
        for ((boundIndex, upperBound) in upperBounds.withIndex()) {
            if (boundIndex > 0) {
                sig.append("&")
            }
            sig.encodeForSignature(upperBound, typeParameterNamer)
        }
    }

    return sig.toString()
}

private fun StringBuilder.encodeForSignature(
        type: KotlinType,
        typeParameterNamer: (TypeParameterDescriptor) -> String
): StringBuilder {
    val declaration = type.constructor.declarationDescriptor!!
    if (declaration is TypeParameterDescriptor) {
        return append(typeParameterNamer(declaration))
    }

    if (type.isSuspendFunctionType) {
        append(DescriptorUtils.getFqName(declaration).asString().replace("kotlin.coroutines.SuspendFunction", "kotlin.SuspendFunction"))
    } else {
        append(DescriptorUtils.getFqName(declaration).asString())
    }

    val parameters = declaration.typeConstructor.parameters
    if (type.arguments.isNotEmpty() && parameters.isNotEmpty()) {
        append("<")
        for ((argument, parameter) in type.arguments.zip(parameters)) {
            if (parameter.index > 0) {
                append(",")
            }
            encodeForSignature(argument, parameter, typeParameterNamer)
        }
        append(">")
    }

    if (type.isMarkedNullable) {
        append("?")
    }

    return this
}

private fun StringBuilder.encodeForSignature(
        projection: TypeProjection,
        parameter: TypeParameterDescriptor,
        typeParameterNamer: (TypeParameterDescriptor) -> String
): StringBuilder {
    return if (projection.isStarProjection) {
        append("*")
    }
    else {
        when (getEffectiveVariance(parameter.variance, projection.projectionKind)) {
            Variance.IN_VARIANCE -> append("-")
            Variance.OUT_VARIANCE -> append("+")
            Variance.INVARIANT -> {}
        }
        encodeForSignature(projection.type, typeParameterNamer)
    }
}

private fun nameTypeParameters(descriptor: DeclarationDescriptor): Map {
    val result = mutableMapOf()
    for ((listIndex, list) in collectTypeParameters(descriptor).withIndex()) {
        for ((indexInList, typeParameter) in list.withIndex()) {
            result[typeParameter] = "$listIndex:$indexInList"
        }
    }
    return result
}

private fun collectTypeParameters(descriptor: DeclarationDescriptor): List> {
    var currentDescriptor: DeclarationDescriptor? = descriptor
    val result = mutableListOf>()
    while (currentDescriptor != null) {
        getOwnTypeParameters(currentDescriptor)?.let { result += it }
        currentDescriptor = if (currentDescriptor is ConstructorDescriptor) {
            currentDescriptor.constructedClass.containingDeclaration
        }
        else {
            currentDescriptor.containingDeclaration
        }
    }
    return result
}

private fun getOwnTypeParameters(descriptor: DeclarationDescriptor): List? =
        when (descriptor) {
            is ClassDescriptor -> descriptor.declaredTypeParameters.filter { !it.isCapturedFromOuterDeclaration }
            is PropertyAccessorDescriptor -> getOwnTypeParameters(descriptor.correspondingProperty)
            is CallableDescriptor -> descriptor.typeParameters.filter { !it.isCapturedFromOuterDeclaration }
            else -> null
        }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy