commonMain.interactive.InteractiveMainUtils.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kommandsamples-jvm Show documentation
Show all versions of kommandsamples-jvm Show documentation
Kotlin DSL for popular CLI commands.
package pl.mareklangiewicz.interactive
import pl.mareklangiewicz.annotations.*
import pl.mareklangiewicz.bad.*
import pl.mareklangiewicz.kground.*
import pl.mareklangiewicz.kground.io.*
import pl.mareklangiewicz.kgroundx.maintenance.*
import pl.mareklangiewicz.kommand.*
import pl.mareklangiewicz.udata.*
import pl.mareklangiewicz.ulog.*
import pl.mareklangiewicz.ulog.hack.*
import pl.mareklangiewicz.usubmit.*
import pl.mareklangiewicz.usubmit.xd.*
/**
* Experimenting directly in kotlin notebooks would be ideal, but the IDE support it's still not great...
* So this fun (called from main fun) allows invoking any code pointed by reference or clipboard (containing reference)
* (see also IntelliJ action: CopyReference)
* Usually it will be from samples/examples/demos, or from gitignored playground, like:
* pl.mareklangiewicz.kommand.demo.MyDemoSamples#getBtop
* pl.mareklangiewicz.kommand.app.Playground#play
* So way we have the IDE support, and later we can C&P working code snippets into notebooks or whateva.
* The gradle kommandapp:run task is set up to run the mainCodeExperiments fun here.
*/
@NotPortableApi
@DelicateApi("API for manual interactive experimentation. Careful because it an easily call ANY code with reflection.")
@ExperimentalApi("Will be removed someday. Temporary solution for running some code parts fast. Like examples/samples.")
suspend fun mainCodeExperiments(args: Array) {
val log = UHackySharedFlowLog { level, data -> "L ${level.symbol} ${data.str(maxLength = 512)}" }
val submit = ZenitySupervisor()
val cli = getSysCLI()
val a0 = args.getOrNull(0).orEmpty()
val a1 = args.getOrNull(1).orEmpty()
val a2 = args.getOrNull(2).orEmpty()
// uctxWithIO(log + submit + cli, dispatcher = null) { // FIXME_later: rethink default dispatcher..
uctxWithIO(log + submit + cli, name = a0) {
when {
args.size == 2 && a0 == "try-code" -> tryInteractivelyCodeRefWithLogging(a1)
args.size == 2 && a0 == "get-user-flag" -> log.i(getUserFlagFullStr(cli, a1))
args.size == 3 && a0 == "set-user-flag" -> setUserFlag(cli, a1, a2.toBoolean())
else -> bad { "Incorrect args. See KommandLine -> InteractiveSamples.kt -> mainCodeExperiments" }
}
}
}
@NotPortableApi
@DelicateApi("API for manual interactive experimentation. Careful because it an easily call ANY code with reflection.")
@ExperimentalApi("Will be removed someday. Temporary solution for running some code parts fast. Like examples/samples.")
suspend fun tryInteractivelyCodeRefWithLogging(reference: String) {
val log = localULog()
try {
log.w("try-code $reference starting")
withLogBadStreams {
tryInteractivelySomethingRef(reference)
log.w("try-code $reference finished")
}
} catch (ex: Exception) {
log.w("try-code $reference failed")
log.exWithTrace(ex)
}
tryInteractivelyOpenLogCache()
}
@OptIn(DelicateApi::class, ExperimentalApi::class) suspend fun tryInteractivelyOpenLogCache() {
val lines = localULogAsOrNull()?.flow?.replayCache ?: return
val fs = localUFileSys()
val submit = localUSubmit()
val notes = fs.pathToTmpNotes.toString()
isInteractiveCodeEnabled() && submit.askIf("Try to open log cache in IDE (in tmp.notes)?") || return
writeFileWithDD(lines, notes).ax()
ideOrGVimOpen(notes).ax()
}
// FIXME: common "UStr" utils with different MPP parametrized exception conversions / string representations.
// (start with moving some of what is in UWidgets to KGround) (see comments in UHackyLog)
@Deprecated("This is temporary fast&dirty impl")
private fun ULog.exWithTrace(ex: Throwable) {
e(ex.toString())
ex.stackTraceToString().let {
e("STACK TRACE:")
it.toList().logEach(this, ULogLevel.ERROR)
// each line have to be logged separately: don't use it.joinToString("\n"), because truncation in logger
}
ex.cause?.let {
e("CAUSE:")
exWithTrace(it)
}
}