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

org.jetbrains.kotlin.resolve.checkers.SealedInheritorInSameModuleChecker.kt Maven / Gradle / Ivy

There is a newer version: 2.0.0
Show newest version
/*
 * Copyright 2010-2020 JetBrains s.r.o. and Kotlin Programming Language contributors.
 * Use of this source code is governed by the Apache 2.0 license that can be found in the license/LICENSE.txt file.
 */

package org.jetbrains.kotlin.resolve.checkers

import org.jetbrains.kotlin.descriptors.ClassDescriptor
import org.jetbrains.kotlin.descriptors.ClassifierDescriptor
import org.jetbrains.kotlin.descriptors.DeclarationDescriptor
import org.jetbrains.kotlin.descriptors.isSealed
import org.jetbrains.kotlin.diagnostics.Errors
import org.jetbrains.kotlin.psi.KtClassOrObject
import org.jetbrains.kotlin.psi.KtDeclaration
import org.jetbrains.kotlin.psi.KtFile
import org.jetbrains.kotlin.resolve.bindingContextUtil.getAbbreviatedTypeOrType
import org.jetbrains.kotlin.resolve.descriptorUtil.module
import org.jetbrains.kotlin.resolve.multiplatform.isCommonSource
import org.jetbrains.kotlin.resolve.source.PsiSourceFile

object SealedInheritorInSameModuleChecker : DeclarationChecker {
    override fun check(declaration: KtDeclaration, descriptor: DeclarationDescriptor, context: DeclarationCheckerContext) {
        if (descriptor !is ClassDescriptor || declaration !is KtClassOrObject) return
        val currentModule = descriptor.module
        for (superTypeListEntry in declaration.superTypeListEntries) {
            val typeReference = superTypeListEntry.typeReference ?: continue
            val superType = typeReference.getAbbreviatedTypeOrType(context.trace.bindingContext)?.unwrap() ?: continue
            val superClass = superType.constructor.declarationDescriptor ?: continue
            if (superClass.isSealed()) {
                /*
                 * If this condition is true then we compile code from CLI mode and class came from
                 *   common source set (in CLI mode we have single ModuleDescriptor for all MPP modules)
                 *
                 * We don't want to analyze classes from common modules, because there are problems with determining
                 *   relation between module of class and module of super class (it's allowed to declare sealed
                 *   inheritors between `expect` class declaration and its actual), so we assume that incorrect
                 *   declarations will be reported while compiling common module to get metadata
                 *
                 *   common               expect sealed class Base
                 *     |                  class A : Base() // OK, A in same module with Base
                 *     |
                 * jvm-js-native          class B : Base() // OK, B inherits `expect` class, not `actual`
                 *     |
                 *   jvm-js               actual sealed class Base
                 *     |                  class C : Base() // OK, C in same module with actual Base
                 *     |
                 *   jvm-js-2             class D : Base() // Error, D not in same module with actual Base
                 *     |
                 *    jvm                 class E : Base() // Error, E not in same module with actual Base
                 *
                 * In this hierarchy error on `D` will be reported in process of compiling `jvm-js-2` module in metadata mode
                 *   (so `jvm-js-2` assumed as "platform" module and others are "common") and won't be reported during
                 *   compilation of `jvm`.
                 *
                 * There is one problem with this condition: if modules hierarchy is a bamboo (no modules with two children)
                 *   then we won't compile intermediate modules in metadata mode, only the last one, so error on class `D`
                 *   won't be reported. See KT-46031
                 */
                if (descriptor.isFromCommonSource && superClass.module == currentModule) {
                    return
                }
                /*
                 * It's allowed to declare inheritors of expect sealed class in any dependent module until actual
                 *   counterpart for this class will be declared. So if super class is resolved to expect sealed
                 *   class then its allowed to declare inheritor
                 */
                if (!(superClass.isExpect || (superClass.module == currentModule && !superClass.isFromCommonSource))) {
                    context.trace.report(Errors.SEALED_INHERITOR_IN_DIFFERENT_MODULE.on(typeReference))
                }
            }
        }
    }

    private val ClassifierDescriptor.isFromCommonSource: Boolean
        get() = ((this.source.containingFile as? PsiSourceFile)?.psiFile as? KtFile)?.isCommonSource ?: false
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy