Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/pgreze/kotlin-process
Kotlin Coroutines friendly way to run an external process
https://github.com/pgreze/kotlin-process
kotlin kotlin-script kotlin-scripting process
Last synced: 13 days ago
JSON representation
Kotlin Coroutines friendly way to run an external process
- Host: GitHub
- URL: https://github.com/pgreze/kotlin-process
- Owner: pgreze
- License: apache-2.0
- Created: 2021-02-05T06:18:41.000Z (almost 4 years ago)
- Default Branch: main
- Last Pushed: 2024-06-27T08:37:37.000Z (7 months ago)
- Last Synced: 2024-06-28T06:14:00.209Z (7 months ago)
- Topics: kotlin, kotlin-script, kotlin-scripting, process
- Language: Kotlin
- Homepage: https://kotlin-process.netlify.app/
- Size: 209 KB
- Stars: 93
- Watchers: 3
- Forks: 4
- Open Issues: 2
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# kotlin-process [![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0) ![Build](https://github.com/pgreze/kotlin-process/workflows/Build/badge.svg) [![codecov](https://codecov.io/gh/pgreze/kotlin-process/branch/main/graph/badge.svg?token=PDyl2T0EEB)](https://codecov.io/gh/pgreze/kotlin-process)
Functional Kotlin friendly way to create external system processes by leveraging:
- Kotlin coroutines
- the powerful but convoluted [ProcessBuilder](https://docs.oracle.com/javase/7/docs/api/java/lang/ProcessBuilder.html)## Installation [![central](https://maven-badges.herokuapp.com/maven-central/com.github.pgreze/kotlin-process/badge.svg?style={style})](https://search.maven.org/artifact/com.github.pgreze/kotlin-process) [![](https://img.shields.io/badge/Java-8-blue)](https://adoptopenjdk.net/) [![](https://img.shields.io/badge/Kotlin-1.6.21-blue)](https://kotlinlang.org/)
```kotlin
repositories {
mavenCentral()
}dependencies {
// Check the 🔝 maven central badge 🔝 for the latest $kotlinProcessVersion
implementation("com.github.pgreze:kotlin-process:$kotlinProcessVersion")
}
```Or in your kotlin script:
```kotlin
@file:DependsOn("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.8.1")
@file:DependsOn("com.github.pgreze:kotlin-process:$kotlinProcessVersion")
```## Usage [![](https://img.shields.io/badge/dokka-read-blue)](https://kotlin-process.netlify.app/)
### Launch a script and consume its results
Starts a program and prints its stdout/stderr outputs to the terminal:
```kotlin
import com.github.pgreze.process.process
import kotlinx.coroutines.runBlockingrunBlocking {
val res = process("echo", "hello world")check(res.resultCode == 0)
// By default process output is displayed in the console.
check(res.output == emptyList())
}
```The next step would probably be to capture the output stream,
in order to process some data from our own-made script:```kotlin
val output = process(
"./my-script.sh", arg1, arg2,// Capture stdout lines to do some operations after
stdout = Redirect.CAPTURE,// Default value: prints to System.err
stderr = Redirect.PRINT,).unwrap() // Fails if the resultCode != 0
// TODO: process the output
println("Success:\n${output.joinToString("\n")}")
```Notice that if you want to capture both stdout and stderr,
there will be no way to differentiate them in the returned output:```kotlin
val res = process(
"./long-and-dangerous.sh", arg1, arg2,// Both streams will be captured,
// preserving their orders but mixing them in the given output.
stdout = Redirect.CAPTURE,
stderr = Redirect.CAPTURE,// Allows to consume line by line without delay the provided output.
consumer = { line -> TODO("process $line") },
)println("Script finished with result=${res.resultCode}")
println("stdout+stderr:\n" + res.output.joinToString("\n"))
```It's also possible to redirect an output stream to a file,
or manually by consuming a Flow instance.```kotlin
import com.github.pgreze.process.Redirect
import com.github.pgreze.process.process
import java.io.File
import java.util.Collections
import kotlinx.coroutines.flow.toList
import kotlinx.coroutines.runBlockingval errLines = Collections.synchronizedList(mutableListOf())
val res = process(
"./my-script.sh", arg1, arg2,// You can save the execution result in a file,
stdout = Redirect.ToFile(File("my-input.txt")),// If you want to handle this stream yourself,
// a Flow instance can be used.
stderr = Redirect.Consume { flow -> flow.toList(errLines) },
)
```The last but not least, you can just silence a stream with Redirect.SILENT 😶
### Control the environment
Several other options are available to control the script environment:
```kotlin
import com.github.pgreze.process.InputSource
import java.io.Fileval res = process(
"./my-script.sh",
// Provides the input as a string, similar to:
// $ echo "hello world" | my-script.sh
stdin = InputSource.fromString("hello world"),// Inject custom environment variables:
env = mapOf("MY_ENV_VARIABLE" to "42"),// Override the working directory:
directory = File("./a/directory"),
)
```There are other ways to provide the process input:
```kotlin
// From a file:
process(
"./my-script.sh",
stdin = InputSource.FromFile(File("my-input.txt")),
)// From an InputStream:
process(
"./my-script.sh",
stdin = InputSource.fromInputStream(myInputStream)),
)// Manually by using the raw OutputStream:
process(
"./my-script.sh",
stdin = InputSource.FromStream { out: OutputStream ->
out.write("hello world\n".toByteArray())
out.flush()
},
)
```## Alternative(s)
1. https://github.com/jakubriegel/kotlin-shell
2. https://github.com/lordcodes/turtle