org.jetbrains.kotlinx.jupyter.repl.impl.JupyterCompilerImpl.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlin-jupyter-kernel Show documentation
Show all versions of kotlin-jupyter-kernel Show documentation
Kotlin Jupyter kernel published as artifact
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.currentKernelVersion
import org.jetbrains.kotlinx.jupyter.exceptions.ReplCompilerException
import org.jetbrains.kotlinx.jupyter.exceptions.ReplException
import org.jetbrains.kotlinx.jupyter.exceptions.getErrors
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.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.util.isError
import kotlin.script.experimental.jvm.util.isIncomplete
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): 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()
override val lastClassLoader: ClassLoader
get() = classes.lastOrNull()?.java?.classLoader
?: basicEvaluationConfiguration[ScriptEvaluationConfiguration.jvm.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
}
}
override fun compileSync(snippet: SourceCode): JupyterCompiler.Result {
when (val resultWithDiagnostics = runBlocking { compiler.compile(snippet, compilationConfig) }) {
is ResultWithDiagnostics.Failure -> throw ReplCompilerException(snippet.text, resultWithDiagnostics)
is ResultWithDiagnostics.Success -> {
val result = resultWithDiagnostics.value
val compiledScript = result.get()
val configWithClassloader = basicEvaluationConfiguration.with {
val lastClassOrNull = classes.lastOrNull()
if (lastClassOrNull != null) {
jvm {
baseClassLoader(lastClassOrNull.java.classLoader)
}
}
}
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)
}
}
}
}
throw ReplException("Impossible situation: this code should be unreachable")
}
}
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)
return when {
result.isIncomplete() -> CheckResult(false)
result.isError() -> throw ReplException(result.getErrors())
else -> CheckResult(true)
}
}
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 = hostConfiguration,
initAnalyzer = { sharedScriptCompilationContext, scopeProcessor ->
ReplCodeAnalyzerBase(
sharedScriptCompilationContext.environment,
implicitsResolutionFilter = scopeProcessor
)
}
)
return JupyterCompilerImpl(
SimpleReplCompiler(
compilationConfiguration[ScriptCompilationConfiguration.hostConfiguration]
?: defaultJvmScriptingHostConfiguration
),
compilationConfiguration,
evaluationConfiguration
)
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy