org.jetbrains.kotlinx.jupyter.repl.impl.JupyterCompilerImpl.kt Maven / Gradle / Ivy
package org.jetbrains.kotlinx.jupyter.repl.impl
import kotlinx.coroutines.runBlocking
import org.jetbrains.kotlin.scripting.compiler.plugin.impl.KJvmReplCompilerBase
import org.jetbrains.kotlin.scripting.compiler.plugin.repl.ReplCodeAnalyzerBase
import org.jetbrains.kotlin.scripting.ide_services.compiler.KJvmReplCompilerWithIdeServices
import org.jetbrains.kotlin.scripting.resolve.KtFileScriptSource
import org.jetbrains.kotlin.scripting.resolve.getScriptCollectedData
import org.jetbrains.kotlinx.jupyter.CheckResult
import org.jetbrains.kotlinx.jupyter.api.Code
import org.jetbrains.kotlinx.jupyter.api.FileAnnotationHandler
import org.jetbrains.kotlinx.jupyter.api.KotlinKernelVersion
import org.jetbrains.kotlinx.jupyter.compiler.util.SourceCodeImpl
import org.jetbrains.kotlinx.jupyter.compiler.util.actualClassLoader
import org.jetbrains.kotlinx.jupyter.config.JupyterCompilingOptions
import org.jetbrains.kotlinx.jupyter.config.currentKernelVersion
import org.jetbrains.kotlinx.jupyter.config.jupyterOptions
import org.jetbrains.kotlinx.jupyter.exceptions.ReplCompilerException
import org.jetbrains.kotlinx.jupyter.exceptions.ReplException
import org.jetbrains.kotlinx.jupyter.exceptions.getErrors
import org.jetbrains.kotlinx.jupyter.util.createCachedFun
import java.util.concurrent.atomic.AtomicInteger
import kotlin.reflect.KClass
import kotlin.script.experimental.api.KotlinType
import kotlin.script.experimental.api.ReplAnalyzerResult
import kotlin.script.experimental.api.ReplCompiler
import kotlin.script.experimental.api.ReplCompleter
import kotlin.script.experimental.api.ResultWithDiagnostics
import kotlin.script.experimental.api.ScriptCompilationConfiguration
import kotlin.script.experimental.api.ScriptConfigurationRefinementContext
import kotlin.script.experimental.api.ScriptDiagnostic
import kotlin.script.experimental.api.ScriptEvaluationConfiguration
import kotlin.script.experimental.api.SourceCode
import kotlin.script.experimental.api.analysisDiagnostics
import kotlin.script.experimental.api.asSuccess
import kotlin.script.experimental.api.defaultImports
import kotlin.script.experimental.api.hostConfiguration
import kotlin.script.experimental.api.refineConfiguration
import kotlin.script.experimental.api.refineConfigurationBeforeCompiling
import kotlin.script.experimental.api.refineOnAnnotations
import kotlin.script.experimental.api.valueOr
import kotlin.script.experimental.api.valueOrNull
import kotlin.script.experimental.api.valueOrThrow
import kotlin.script.experimental.api.with
import kotlin.script.experimental.host.ScriptingHostConfiguration
import kotlin.script.experimental.jvm.baseClassLoader
import kotlin.script.experimental.jvm.defaultJvmScriptingHostConfiguration
import kotlin.script.experimental.jvm.impl.KJvmCompiledScript
import kotlin.script.experimental.jvm.impl.getOrCreateActualClassloader
import kotlin.script.experimental.jvm.jvm
import kotlin.script.experimental.jvm.lastSnippetClassLoader
import kotlin.script.experimental.jvm.util.toSourceCodePosition
import kotlin.script.experimental.util.LinkedSnippet
interface JupyterCompiler {
val version: KotlinKernelVersion
val numberOfSnippets: Int
val previousScriptsClasses: List>
val lastKClass: KClass<*>
val lastClassLoader: ClassLoader
fun nextCounter(): Int
fun updateCompilationConfig(body: ScriptCompilationConfiguration.Builder.() -> Unit)
fun updateCompilationConfigOnAnnotation(handler: FileAnnotationHandler, callback: (ScriptConfigurationRefinementContext) -> ResultWithDiagnostics)
fun compileSync(snippet: SourceCode, options: JupyterCompilingOptions): Result
data class Result(
val snippet: LinkedSnippet,
val newEvaluationConfiguration: ScriptEvaluationConfiguration,
)
}
interface JupyterCompilerWithCompletion : JupyterCompiler {
val completer: ReplCompleter
fun checkComplete(code: Code): CheckResult
fun listErrors(code: Code): Sequence
companion object {
fun create(
compilationConfiguration: ScriptCompilationConfiguration,
evaluationConfiguration: ScriptEvaluationConfiguration,
): JupyterCompilerWithCompletion {
return JupyterCompilerWithCompletionImpl(
KJvmReplCompilerWithIdeServices(
compilationConfiguration[ScriptCompilationConfiguration.hostConfiguration]
?: defaultJvmScriptingHostConfiguration,
),
compilationConfiguration,
evaluationConfiguration,
)
}
}
}
open class JupyterCompilerImpl>(
protected val compiler: CompilerT,
initialCompilationConfig: ScriptCompilationConfiguration,
private val basicEvaluationConfiguration: ScriptEvaluationConfiguration,
) : JupyterCompiler {
private val executionCounter = AtomicInteger()
private val classes = mutableListOf>()
private val refinementCallbacks = mutableListOf<(ScriptConfigurationRefinementContext) -> ResultWithDiagnostics>()
protected val compilationConfig: ScriptCompilationConfiguration = initialCompilationConfig.with {
refineConfiguration {
val handlers = initialCompilationConfig[ScriptCompilationConfiguration.refineConfigurationBeforeCompiling].orEmpty()
handlers.forEach { beforeCompiling(it.handler) }
beforeCompiling(::updateConfig)
}
}
override val version: KotlinKernelVersion = currentKernelVersion
override val numberOfSnippets: Int
get() = classes.size
override val previousScriptsClasses: List>
get() = classes
override val lastKClass: KClass<*>
get() = classes.last()
private val _baseClassLoader: ClassLoader
get() = basicEvaluationConfiguration[ScriptEvaluationConfiguration.jvm.baseClassLoader]!!
override val lastClassLoader: ClassLoader
get() = classes.lastOrNull()?.java?.classLoader ?: _baseClassLoader
override fun nextCounter() = executionCounter.getAndIncrement()
override fun updateCompilationConfig(body: ScriptCompilationConfiguration.Builder.() -> Unit) {
refinementCallbacks.add { context ->
context.compilationConfiguration.with(body).asSuccess()
}
}
override fun updateCompilationConfigOnAnnotation(handler: FileAnnotationHandler, callback: (ScriptConfigurationRefinementContext) -> ResultWithDiagnostics) {
refinementCallbacks.add { context ->
val ktFile = (context.script as KtFileScriptSource).ktFile
val withImport = context.compilationConfiguration.with {
defaultImports(handler.annotation.java.name)
refineConfiguration {
onAnnotations(KotlinType(handler.annotation.qualifiedName!!)) {
callback(it)
}
}
}
val collectedData = getScriptCollectedData(ktFile, withImport, ktFile.project, lastClassLoader)
withImport.refineOnAnnotations(context.script, collectedData)
}
}
private fun updateConfig(context: ScriptConfigurationRefinementContext): ResultWithDiagnostics {
return refinementCallbacks.fold(context.compilationConfiguration.asSuccess()) { config: ResultWithDiagnostics, callback ->
config.valueOrNull()?.let { conf ->
callback(
ScriptConfigurationRefinementContext(
context.script,
conf,
context.collectedData,
),
)
} ?: config
}
}
private val getCompilationConfiguration = createCachedFun { options: JupyterCompilingOptions ->
compilationConfig.with { jupyterOptions(options) }
}
override fun compileSync(snippet: SourceCode, options: JupyterCompilingOptions): JupyterCompiler.Result {
val compilationConfigWithJupyterOptions = getCompilationConfiguration(options)
when (val resultWithDiagnostics = runBlocking { compiler.compile(snippet, compilationConfigWithJupyterOptions) }) {
is ResultWithDiagnostics.Failure -> throw ReplCompilerException(snippet.text, resultWithDiagnostics)
is ResultWithDiagnostics.Success -> {
val result = resultWithDiagnostics.value
val compiledScript = result.get()
val configWithClassloader = basicEvaluationConfiguration.with {
jvm {
lastSnippetClassLoader(lastClassLoader)
baseClassLoader(_baseClassLoader.parent)
}
}
val classLoader = compiledScript.getOrCreateActualClassloader(configWithClassloader)
val newEvaluationConfiguration = configWithClassloader.with {
jvm {
actualClassLoader(classLoader)
}
}
when (val kClassWithDiagnostics = runBlocking { compiledScript.getClass(newEvaluationConfiguration) }) {
is ResultWithDiagnostics.Failure -> throw ReplCompilerException(snippet.text, kClassWithDiagnostics)
is ResultWithDiagnostics.Success -> {
val kClass = kClassWithDiagnostics.value
classes.add(kClass)
return JupyterCompiler.Result(result, newEvaluationConfiguration)
}
}
}
}
}
}
class JupyterCompilerWithCompletionImpl(
compiler: KJvmReplCompilerWithIdeServices,
compilationConfig: ScriptCompilationConfiguration,
evaluationConfig: ScriptEvaluationConfiguration,
) : JupyterCompilerImpl(compiler, compilationConfig, evaluationConfig),
JupyterCompilerWithCompletion {
override val completer: ReplCompleter
get() = compiler
override fun checkComplete(
code: Code,
): CheckResult {
val result = analyze(code)
val analysisResult = result.valueOr { throw ReplException(result.getErrors()) }
val diagnostics = analysisResult[ReplAnalyzerResult.analysisDiagnostics]!!
val isComplete = diagnostics.none { it.code == ScriptDiagnostic.incompleteCode }
return CheckResult(isComplete)
}
private fun analyze(
code: Code,
): ResultWithDiagnostics {
val snippet = SourceCodeImpl(nextCounter(), code)
return runBlocking {
compiler.analyze(
snippet,
0.toSourceCodePosition(snippet),
compilationConfig,
)
}
}
override fun listErrors(code: Code): Sequence {
val result = analyze(code).valueOrThrow()
return result[ReplAnalyzerResult.analysisDiagnostics]!!
}
}
fun getSimpleCompiler(
compilationConfiguration: ScriptCompilationConfiguration,
evaluationConfiguration: ScriptEvaluationConfiguration,
): JupyterCompiler {
class SimpleReplCompiler(hostConfiguration: ScriptingHostConfiguration) :
KJvmReplCompilerBase(hostConfiguration)
return JupyterCompilerImpl(
SimpleReplCompiler(
compilationConfiguration[ScriptCompilationConfiguration.hostConfiguration]
?: defaultJvmScriptingHostConfiguration,
),
compilationConfiguration,
evaluationConfiguration,
)
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy