org.jetbrains.kotlinx.jupyter.ikotlin.kt Maven / Gradle / Ivy
package org.jetbrains.kotlinx.jupyter
import org.jetbrains.kotlinx.jupyter.api.libraries.JupyterSocketType
import org.jetbrains.kotlinx.jupyter.api.libraries.rawMessageCallback
import org.jetbrains.kotlinx.jupyter.libraries.EmptyResolutionInfoProvider
import org.jetbrains.kotlinx.jupyter.libraries.ResolutionInfoProvider
import org.jetbrains.kotlinx.jupyter.libraries.getDefaultClasspathResolutionInfoProvider
import org.jetbrains.kotlinx.jupyter.messaging.CommManagerImpl
import org.jetbrains.kotlinx.jupyter.messaging.JupyterConnectionInternal
import org.jetbrains.kotlinx.jupyter.messaging.controlMessagesHandler
import org.jetbrains.kotlinx.jupyter.messaging.shellMessagesHandler
import org.jetbrains.kotlinx.jupyter.repl.creating.DefaultReplFactory
import org.jetbrains.kotlinx.jupyter.startup.KernelArgs
import org.jetbrains.kotlinx.jupyter.startup.KernelConfig
import org.jetbrains.kotlinx.jupyter.startup.getConfig
import java.io.File
import java.util.concurrent.atomic.AtomicLong
import kotlin.concurrent.thread
import kotlin.script.experimental.jvm.util.classpathFromClassloader
val iKotlinClass: Class<*> = object {}::class.java.enclosingClass
private fun parseCommandLine(vararg args: String): KernelArgs {
var cfgFile: File? = null
var classpath: List? = null
var homeDir: File? = null
var debugPort: Int? = null
var clientType: String? = null
var jvmTargetForSnippets: String? = null
args.forEach { arg ->
when {
arg.startsWith("-cp=") || arg.startsWith("-classpath=") -> {
classpath?.let {
throw IllegalArgumentException("classpath already set to ${it.joinToString(File.pathSeparator)}")
}
classpath = arg.substringAfter('=').split(File.pathSeparator).map { File(it) }
}
arg.startsWith("-home=") -> {
homeDir = File(arg.substringAfter('='))
}
arg.startsWith("-debugPort=") -> {
debugPort = arg.substringAfter('=').toInt()
}
arg.startsWith("-client=") -> {
clientType = arg.substringAfter('=')
}
arg.startsWith("-jvmTarget") -> {
jvmTargetForSnippets = arg.substringAfter('=')
}
else -> {
cfgFile?.let { throw IllegalArgumentException("config file already set to $it") }
cfgFile = File(arg)
}
}
}
val cfgFileValue = cfgFile ?: throw IllegalArgumentException("config file is not provided")
if (!cfgFileValue.exists() || !cfgFileValue.isFile) throw IllegalArgumentException("invalid config file $cfgFileValue")
return KernelArgs(cfgFileValue, classpath ?: emptyList(), homeDir, debugPort, clientType, jvmTargetForSnippets)
}
fun printClassPath() {
val cl = ClassLoader.getSystemClassLoader()
val cp = classpathFromClassloader(cl)
if (cp != null) {
log.info("Current classpath: " + cp.joinToString())
}
}
fun main(vararg args: String) {
try {
log.info("Kernel args: " + args.joinToString { it })
val kernelArgs = parseCommandLine(*args)
val libraryInfoProvider = getDefaultClasspathResolutionInfoProvider()
val kernelConfig = kernelArgs.getConfig()
val replConfig = ReplConfig.create(libraryInfoProvider, kernelArgs.homeDir)
val runtimeProperties = createRuntimeProperties(kernelConfig)
kernelServer(kernelConfig, replConfig, runtimeProperties)
} catch (e: Exception) {
log.error("exception running kernel with args: \"${args.joinToString()}\"", e)
}
}
/**
* This function is to be run in projects which use kernel as a library,
* so we don't have a big need in covering it with tests
*
* The expected use case for this function is embedding into a Java application that doesn't necessarily support extensions written in Kotlin
* The signature of this function should thus be simple, and e.g. allow resolutionInfoProvider to be null instead of having to pass EmptyResolutionInfoProvider
* because EmptyResolutionInfoProvider is a Kotlin singleton object, and it takes a while to understand how to use it from Java code.
*/
@Suppress("unused")
fun embedKernel(cfgFile: File, resolutionInfoProvider: ResolutionInfoProvider?, scriptReceivers: List? = null) {
val cp = System.getProperty("java.class.path").split(File.pathSeparator).toTypedArray().map { File(it) }
val kernelConfig = KernelArgs(cfgFile, cp, null, null, null, null).getConfig()
val replConfig = ReplConfig.create(
resolutionInfoProvider ?: EmptyResolutionInfoProvider,
null,
true,
)
kernelServer(kernelConfig, replConfig, scriptReceivers = scriptReceivers ?: emptyList())
}
fun kernelServer(kernelConfig: KernelConfig, replConfig: ReplConfig, runtimeProperties: ReplRuntimeProperties = defaultRuntimeProperties, scriptReceivers: List = emptyList()) {
log.info("Starting server with config: $kernelConfig")
JupyterConnectionImpl(kernelConfig).use { conn: JupyterConnectionInternal ->
printClassPath()
log.info("Begin listening for events")
val executionCount = AtomicLong(1)
val socketManager = conn.socketManager
val commManager = CommManagerImpl(conn)
val repl = DefaultReplFactory(kernelConfig, replConfig, runtimeProperties, scriptReceivers, conn, commManager).createRepl()
val mainThread = Thread.currentThread()
fun socketLoop(
interruptedMessage: String,
vararg threadsToInterrupt: Thread,
loopBody: () -> Unit,
) {
while (true) {
try {
loopBody()
} catch (e: InterruptedException) {
log.debug(interruptedMessage)
threadsToInterrupt.forEach { it.interrupt() }
break
}
}
}
conn.addMessageCallback(
rawMessageCallback(JupyterSocketType.CONTROL, null) { rawMessage ->
conn.controlMessagesHandler(rawMessage, repl)
},
)
conn.addMessageCallback(
rawMessageCallback(JupyterSocketType.SHELL, null) { rawMessage ->
conn.messageFactory.updateSessionInfo(rawMessage)
conn.shellMessagesHandler(rawMessage, repl, commManager, executionCount)
},
)
val controlThread = thread {
socketLoop("Control: Interrupted", mainThread) {
socketManager.control.runCallbacksOnMessage()
}
}
val hbThread = thread {
socketLoop("Heartbeat: Interrupted", mainThread) {
socketManager.heartbeat.onData { send(it) }
}
}
socketLoop("Main: Interrupted", controlThread, hbThread) {
socketManager.shell.runCallbacksOnMessage()
}
try {
controlThread.join()
hbThread.join()
} catch (_: InterruptedException) {
}
log.info("Shutdown server")
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy