Skip to content

Latest commit

 

History

History
202 lines (142 loc) · 6.17 KB

README.md

File metadata and controls

202 lines (142 loc) · 6.17 KB

Ktast Java CI

ktast is a simple library that represents Kotlin source code as an Abstract Syntax Tree (AST), allowing for node manipulation. It can parse Kotlin source code into an AST, make modifications, and write it back to the source code. The features of the ktast are as follows:

  • It supports the latest version of Kotlin syntax
  • AST nodes are represented as immutable data classes, and the copy method can be utilized
  • It can handle whitespaces, comments, semicolons and trailing commas while manipulating AST nodes

It includes the following components:

Related work

Ktast is a fork of Kastree. Unfortunately Kastree is currently not being actively developed. We are grateful to Chad Retz and contributors for the great work. Without their effort, we cannot build this library.

Another kotlin AST parsing library is kotlinx.ast. It does not use the Kotlin Compiler, but uses ANTLR and official Kotlin Grammar. Currently, it seems that the library's summary AST classes are limited.

ktcodeshift is a toolkit for running codemods over multiple Kotlin files built on top of ktast. It is useful when modifying Kotlin source codes.

Usage

Getting started

Ktast is available on JitPack. Add the following configurations to your Gradle build settings:

repositories {
    maven { setUrl("https://jitpack.io") }
}

dependencies {
    implementation("com.github.orangain.ktast:ast-psi:0.9.3")
}

The library ast-psi transitively depends on the Kotlin compiler.

While the parser only works from JVM projects, the AST itself (and writers/visitors) can be used from other multiplatform projects. If you need the AST only, instead use:

dependencies {
    implementation("com.github.orangain.ktast:ast:0.9.3")
}

Documentation

API document is available at:

https://orangain.github.io/ktast/latest/api/

Examples

Examples below are simple Kotlin scripts.

Parsing code

In this example, we use the Parser, which is a wrapper around the Kotlin compiler's parser:

import ktast.ast.psi.Parser

val code = """
    package foo

    fun bar() {
        // Print hello
        println("Hello, World!")
    }

    fun baz() = println("Hello, again!")
""".trimIndent()
// Call the parser with the code
val file = Parser.parseFile(code)

The file variable is now a ktast.ast.Node.KotlinFile. Each AST nodes have blank line and comment information. If you don't need them, you can pass a Converter instance to the constructor argument of the Parser instead:

import ktast.ast.psi.Parser
import ktast.ast.psi.Converter

// ...
val fileWithoutExtras = Parser(Converter()).parseFile(code)

Writing code

To write the code from the node created above, simply use the Writer:

import ktast.ast.Writer

// ...

println(Writer.write(file))

This outputs the following code, which is exactly the same code as the input. This is because the AST nodes have blank line and comment information.

package foo

fun bar() {
    // Print hello
    println("Hello, World!")
}

fun baz() = println("Hello, again!")

Visiting nodes

To get all strings from the file, we can use the Visitor:

import ktast.ast.Node
import ktast.ast.Visitor

// ...

val strings = mutableListOf<String>()
Visitor.traverse(file) { path ->
    val node = path.node
    if (node is Node.Expression.StringLiteralExpression.LiteralStringEntry) {
        strings.add(node.text)
    }
}
// Prints [Hello, World!, Hello, again!]
println(strings)

The parameter of the lambda is a NodePath object that has node and parent NodePath.

Modifying nodes

To modify the file, we can use the MutableVisitor. The following code will change "Hello, World!" and "Hello, again!" to "Howdy, World!" and "Howdy, again":

import ktast.ast.MutableVisitor

// ...

val newFile = MutableVisitor.traverse(file) { path ->
    val node = path.node
    if (node !is Node.Expression.StringLiteralExpression.LiteralStringEntry) node
    else node.copy(text = node.text.replace("Hello", "Howdy"))
}

Writer.write(newFile)

Now newFile is a transformed version of file. As before, the parameter of the lambda is a NodePath object. The output will be the following code:

package foo

fun bar() {
    // Print hello
    println("Howdy, World!")
}

fun baz() = println("Howdy, again!")

Note that the comments and blank lines are preserved.

Running tests

The tests rely on a checked out version of the Kotlin source repository since it uses the test data there to build a corpus to test against. The path to the base of the repo needs to be set as the KOTLIN_REPO environment variable. For example, run:

KOTLIN_REPO=~/kotlin ./gradlew :ast-psi:test

This will ignore all Kotlin files with expected parse errors and only test against the ones that are valid (178 as of this writing). The test parses the Kotlin code into this AST, then re-writes this AST, then re-parses what was just written and confirms it matches the original AST field for field.