org.jetbrains.kotlin.container.Container.kt Maven / Gradle / Ivy
/*
* Copyright 2010-2015 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.container
import java.io.Closeable
import java.io.PrintStream
import java.lang.reflect.ParameterizedType
import java.lang.reflect.Type
import java.lang.reflect.WildcardType
class ContainerConsistencyException(message: String) : Exception(message)
interface ComponentContainer {
val containerId: String
fun createResolveContext(requestingDescriptor: ValueDescriptor): ValueResolveContext
}
interface ComponentProvider {
fun resolve(request: Type): ValueDescriptor?
fun create(request: Class): T
}
object DynamicComponentDescriptor : ValueDescriptor {
override fun getValue(): Any = throw UnsupportedOperationException()
override fun toString(): String = "Dynamic"
}
class StorageComponentContainer(
private val id: String,
parent: StorageComponentContainer? = null
) : ComponentContainer, ComponentProvider, Closeable {
val unknownContext: ComponentResolveContext by lazy {
val parentContext = parent?.let { ComponentResolveContext(it, DynamicComponentDescriptor) }
ComponentResolveContext(this, DynamicComponentDescriptor, parentContext)
}
private val componentStorage: ComponentStorage = ComponentStorage(id, parent?.componentStorage)
override fun createResolveContext(requestingDescriptor: ValueDescriptor): ValueResolveContext {
if (requestingDescriptor == DynamicComponentDescriptor) // cache unknown component descriptor
return unknownContext
return ComponentResolveContext(this, requestingDescriptor)
}
fun compose(): StorageComponentContainer {
componentStorage.compose(unknownContext)
return this
}
fun dump(printer: PrintStream) {
componentStorage.dump(printer)
}
override fun close() = componentStorage.dispose()
fun resolve(request: Type, context: ValueResolveContext): ValueDescriptor? {
return componentStorage.resolve(request, context) ?: resolveIterable(request, context)
}
override fun resolve(request: Type): ValueDescriptor? {
return resolve(request, unknownContext)
}
private fun resolveIterable(request: Type, context: ValueResolveContext): ValueDescriptor? {
if (request !is ParameterizedType) return null
val rawType = request.rawType
if (rawType != Iterable::class.java) return null
val typeArguments = request.actualTypeArguments
if (typeArguments.size != 1) return null
val iterableTypeArgument = typeArguments[0]
val iterableType = when (iterableTypeArgument) {
is WildcardType -> {
val upperBounds = iterableTypeArgument.upperBounds
if (upperBounds.size != 1) return null
upperBounds[0]
}
is Class<*> -> iterableTypeArgument
is ParameterizedType -> iterableTypeArgument
else -> return null
}
return IterableDescriptor(componentStorage.resolveMultiple(iterableType, context))
}
fun resolveMultiple(request: Class<*>, context: ValueResolveContext = unknownContext): Iterable {
return componentStorage.resolveMultiple(request, context)
}
internal fun registerDescriptors(descriptors: List): StorageComponentContainer {
componentStorage.registerDescriptors(unknownContext, descriptors)
return this
}
internal fun registerClashResolvers(resolvers: List>): StorageComponentContainer {
componentStorage.registerClashResolvers(resolvers)
return this
}
override fun create(request: Class): T {
val constructorBinding = request.bindToConstructor(containerId, unknownContext)
val args = constructorBinding.argumentDescriptors.map { it.getValue() }.toTypedArray()
return runWithUnwrappingInvocationException {
@Suppress("UNCHECKED_CAST")
constructorBinding.constructor.newInstance(*args) as T
}
}
override val containerId
get() = "Container: $id"
override fun toString() = containerId
}
fun StorageComponentContainer.registerSingleton(klass: Class<*>): StorageComponentContainer {
return registerDescriptors(listOf(SingletonTypeComponentDescriptor(this, klass)))
}
fun StorageComponentContainer.registerInstance(instance: Any): StorageComponentContainer {
return registerDescriptors(listOf(InstanceComponentDescriptor(instance)))
}
inline fun StorageComponentContainer.resolve(context: ValueResolveContext = unknownContext): ValueDescriptor? {
return resolve(T::class.java, context)
}
inline fun StorageComponentContainer.resolveMultiple(context: ValueResolveContext = unknownContext): Iterable {
return resolveMultiple(T::class.java, context)
}
© 2015 - 2024 Weber Informatics LLC | Privacy Policy