SW 개발

Kotlin, 쉘 스크립트 작성하기

지단로보트 2024. 3. 20. 17:57

개요

  • Bash는 운영 환경에서 시스템 엔지니어 및 개발자들에게 있어 가장 널리 사용되고 익숙한 스크립트 인터프리터이지만 동시에 가장 불편한 스크립트 언어를 제공한다. 수년간 개발 진영은 Type-Safe를 보장하는 언어가 생산성과 유지보수에 있어 강세를 보이면서, 프론트엔드에서는 TypeScript가, 백엔드에서는 Kotlin이 점유율을 높이고 있다. 그렇다면 Kotlin으로 쉘 스크립트를 작성하면 어떨까? 이미 Node.js는 쉘 스크립트에 널리 사용되고 있다. 이번 글에서는 Kotlin으로 쉘 스크립트를 작성하는 방법을 설명하자고 한다.

kscript 설치

  • kscriptKotlin 언어로 쉘 스크립트를 작성하게 해주는 유용한 보조 도구이다. JVM의 단점을 극복하기 위한 스크립트 캐시 등 여러 유용한 도구를 제공한다. 아래와 같이 설치한다.
# SDKMAN 설치
$ curl -s "https://get.sdkman.io" | bash
$ source "$HOME/.sdkman/bin/sdkman-init.sh"

$ nano $HOME/.bash_profile
export SDKMAN_DIR="$HOME/.sdkman"
[[ -s "$HOME/.sdkman/bin/sdkman-init.sh" ]] && source "$HOME/.sdkman/bin/sdkman-init.sh"

# kscript 설치
$ sdk install kotlin
$ sdk install kscript

# 설치된 버전 확인
$ kscript -v
Version   : 4.2.3
Build     : 2023-07-22T13:06:02.327407526Z
Kotlin    : 1.9.23-release-779
Java      : JRE 21.0.1+12

kscript 예제 작성

  • 이제 스크립트를 작성해 볼 차례이다.
# 스크립트 작성
$ nano hello.kts
#!/usr/bin/env kscript
println("Hello, World!")

# 스크립트에 실행 권한 부여
$ chmod +x hello.kts

# 스크립트 실행
$ ./hello.kts
Hello, World!

# IntelliJ IDEA로 편집
$ kscript --idea hello.kts

라이브러리 추가 예제

  • 스크립트의 시작 부분에 @file:DependsOn()를 사용하면 Maven 저장소의 원격 라이브러리를 사용할 수 있다. 아래는 TSID 클래스를 외부 라이브러리에서 임포트하여 사용한 예제이다.
$ nano tsid.kts
#!/usr/bin/env kscript
@file:DependsOn("io.hypersistence:hypersistence-utils-hibernate-60:3.7.3")

import io.hypersistence.tsid.TSID

val tsid = TSID.fast()
println(tsid.toLong())
println(tsid.toString())
println(tsid.toLowerCase())
println(tsid.instant)

외부 명령어 실행 예제

  • 아래와 같이 스크립트를 작성하면 스크립트 내에서 외부 명령어를 실행할 수 있다.
#!/usr/bin/env kscript
import java.io.File

when ("ls -al".exec()) {
    0 -> println("SUCCESS")
    else -> {
        println("FAILED")
        kotlin.system.exitProcess(1) 
    }
}

// 실행 결과를 반환 (0:성공, 나머지:실패)
fun String.exec(cwd: File? = null): Int {
    return ProcessBuilder(*split(" ").toTypedArray())
            .redirectErrorStream(true)
            .inheritIO()
            .directory(cwd)
            .start()
            .waitFor()
}

// 실행 문자열을 반환, 실패시 IOException 발생
fun String.exec(cwd: File? = null): String? {
    val parts = this.split("\\s".toRegex())
    val proc = ProcessBuilder(*parts.toTypedArray())
            .directory(cwd)
            .redirectOutput(ProcessBuilder.Redirect.PIPE)
            .redirectError(ProcessBuilder.Redirect.PIPE)
            .start()
    proc.waitFor()
    return proc.inputStream.bufferedReader().readText()
}

참고 글