main.de.jensklingenberg.ktorfit.CreateFuncTransformer.kt Maven / Gradle / Ivy
package de.jensklingenberg.ktorfit
import org.jetbrains.kotlin.backend.common.IrElementTransformerVoidWithContext
import org.jetbrains.kotlin.backend.common.extensions.IrPluginContext
import org.jetbrains.kotlin.ir.expressions.IrCall
import org.jetbrains.kotlin.ir.expressions.IrExpression
import org.jetbrains.kotlin.ir.expressions.impl.IrConstructorCallImpl
import org.jetbrains.kotlin.ir.types.classFqName
import org.jetbrains.kotlin.ir.types.defaultType
import org.jetbrains.kotlin.ir.types.impl.originalKotlinType
import org.jetbrains.kotlin.ir.util.constructors
import org.jetbrains.kotlin.ir.util.isInterface
import org.jetbrains.kotlin.name.ClassId
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.Name
/**
* Transform exampleKtorfit.create() to exampleKtorfit.create(_TestApiImpl())
*/
class CreateFuncTransformer(
private val pluginContext: IrPluginContext,
private val debugLogger: DebugLogger
) :
IrElementTransformerVoidWithContext() {
companion object {
fun ERROR_TYPE_ARGUMENT_NOT_INTERFACE(implName: String)=
"create<${implName}> argument is not supported. Type argument needs to be an interface"
fun ERROR_IMPL_NOT_FOUND(implName: String) =
"_${implName}Impl() not found, did you apply the Ksp Ktorfit plugin?"
private const val KTORFIT_PACKAGE = "de.jensklingenberg.ktorfit.Ktorfit"
private const val KTORFIT_CREATE = "create"
}
override fun visitExpression(expression: IrExpression): IrExpression {
//Find exampleKtorfit.create()
(expression as? IrCall)?.let { irCall ->
if (irCall.typeArgumentsCount > 0) {
if (!expression.symbol.owner.symbol.toString().contains(KTORFIT_PACKAGE)) {
return expression
}
if (expression.symbol.owner.name.asString() != KTORFIT_CREATE) {
return expression
}
if (expression.getValueArgument(0) != null) {
return expression
}
//Get T from create()
val argumentType = irCall.getTypeArgument(0) ?: return expression
val classFqName = argumentType.classFqName
if (!argumentType.isInterface()) throw IllegalStateException(ERROR_TYPE_ARGUMENT_NOT_INTERFACE(argumentType.originalKotlinType.toString()))
if (classFqName == null) {
throw IllegalStateException(ERROR_IMPL_NOT_FOUND(argumentType.originalKotlinType.toString()))
}
val packageName = classFqName.packageName
val className = classFqName.shortName().toString()
//Find the class _TestApiImpl
val implClassSymbol = pluginContext.referenceClass(
ClassId(
FqName(packageName),
Name.identifier("_$className" + "Impl")
)
) ?: throw IllegalStateException(ERROR_IMPL_NOT_FOUND(argumentType.originalKotlinType.toString()))
val newConstructor = implClassSymbol.constructors.first()
//Create the constructor call for _ExampleApiImpl()
val newCall = IrConstructorCallImpl(
0,
0,
type = implClassSymbol.defaultType,
symbol = newConstructor,
0,
0,
1,
null
).apply {
this.putValueArgument(0, expression.dispatchReceiver)
}
//Set _ExampleApiImpl() as argument for create()
irCall.putValueArgument(0, newCall)
debugLogger.log(
"Transformed " + argumentType.originalKotlinType.toString() + " to _$className" + "Impl"
)
return super.visitExpression(irCall)
}
}
return super.visitExpression(expression)
}
}
private val FqName?.packageName: String
get() {
return this.toString().substringBeforeLast(".")
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy