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