Skip to content

Gradle Plugin: KDoc / JavaDoc Preprocessor that can produce modified sources

License

Notifications You must be signed in to change notification settings

Jolanrensen/docProcessorGradlePlugin

Repository files navigation

Maven metadata URL

KDoc / JavaDoc Preprocessor Gradle Plugin (Beta)

This Gradle plugin allows you to preprocess your KDoc / JavaDoc comments with custom preprocessors and obtain modified sources.

These preprocessors can be used to add custom tags to your KDoc / JavaDoc comments or change the entirety of the comment. This is not a Dokka plugin, meaning you can actually get a sources.jar file with the modified comments instead of just having the comments modified in a javadoc.jar or a Dokka HTML website.

Note: {@inline tags} work in KDoc comments too! Plus, {@tags {@inside tags}} are supported too.

Example

What you write:

Click to see the source

What you get:

Click to see the source

Used By

This plugin is used by Kotlin DataFrame, to make it possible to document and update the wide range of function overloads present in the library due to its DSL-like nature.

Let me know if you're using it in your project too!

Preprocessors

Preprocessors are run one at a time, in order, on all KDoc / JavaDoc comments in the sources. If a preprocessor is a tag processor, it will process only its own tags in the following order:

  • Inline tags
    • depth-first
    • top-to-bottom
    • left-to-right
  • Block tags
    • top-to-bottom

Included preprocessors are:

Description Name
@include tag to include other documentation into your KDoc / JavaDoc.
Used like @include [Reference.To.Element].
INCLUDE_DOC_PROCESSOR
@includeFile tag to include the entire content of a file into your KDoc / JavaDoc.
Used like @includeFile (./path/to/file).
INCLUDE_FILE_DOC_PROCESSOR
@set / @get (or $) tags to define and retrieve variables within a KDoc / JavaDoc. Powerful in combination with @include.
Used like @set KEY some content, @get KEY some default.
Shortcuts for {@get .} are $KEY, $KEY=default, ${KEY}, and ${KEY=some default}.
ARG_DOC_PROCESSOR
@comment tag to comment out parts of your modified KDoc / JavaDoc.
Used like @comment Some comment text.
COMMENT_DOC_PROCESSOR
@sample / @sampleNoComments tags to include code samples into your KDoc / JavaDoc.
Used like @sample [Reference.To.Element].
If present, only code in between // SampleStart and // SampleEnd is taken. @sampleNoComments excludes KDoc / JavaDoc from the sample.
SAMPLE_DOC_PROCESSOR
@exportAsHtmlStart / @exportAsHtmlEnd to mark a range of KDoc for the @ExportAsHtml annotation. EXPORT_AS_HTML_DOC_PROCESSOR
A processor that removes all escape characters ("\") from your KDoc / JavaDoc comments. REMOVE_ESCAPE_CHARS_PROCESSOR
A processor that removes all KDoc / JavaDoc comments. NO_DOC_PROCESSOR
A processor that adds a /** TODO */ comment wherever there is no KDoc / JavaDoc comment. TODO_DOC_PROCESSOR

Of course, you can also try to make your own preprocessor (see Custom Processors). For instance, you could make a processor that makes all KDoc / JavaDoc comments uppercase, a tag processor that automatically inserts URLs to your website, or simply a processor that produces errors or warnings for incorrect doc usage.

The sky is the limit :).

@ExcludeFromSources annotation

If you want to exclude any annotatable element from the sources.jar. Create an annotation class named exactly "ExcludeFromSources" (you can copy the code from here) and annotate the elements you want to exclude with it. This is especially useful for "temporary" documentation interfaces, only there to provide documentation for other elements.

@ExportAsHtml annotation

To export a KDoc comment as HTML, you can use the @ExportAsHtml annotation. Create an annotation class named exactly "ExportAsHtml" and add the arguments theme: Boolean and stripReferences: Boolean (default both to true) (you can copy the code from here). Then, add the annotation to the element you want to export as HTML.

Inside the KDoc comment, you can mark a range of text to be exported as HTML by using the optional @exportAsHtmlStart and @exportAsHtmlEnd tags.

In the Gradle task the HTML will be generated in the folder specified in the exportAsHtml block of the ProcessDocTask (see below). In the IntelliJ plugin, a temporary file will be created that you can open in the browser by clicking on link at the bottom of the rendered KDoc.

How to get it

From Gradle Plugins

In your project's settings.gradle.kts or build.gradle add:

pluginManagement {
    repositories {
        ..
        gradlePluginPortal()
    }
}

In build.gradle.kts or build.gradle add id("nl.jolanrensen.docProcessor") version "{ VERSION }" to plugins { .. }.

From sources

Clone the project and run ./gradlew publishToMavenLocal in the source folder.

In your project's settings.gradle.kts or settings.gradle add:

pluginManagement {
    repositories {
        ..
        mavenLocal()
    }
}

In build.gradle.kts or build.gradle add id("nl.jolanrensen.docProcessor") version "{ VERSION }" to plugins { .. }.

How to use

Say you want to create a task that will run when you're making a sources Jar such that the modified files appear in the Jar:

build.gradle.kts:

import nl.jolanrensen.docProcessor.gradle.*
import nl.jolanrensen.docProcessor.defaultProcessors.*
import org.gradle.jvm.tasks.Jar

..

plugins {
    id("nl.jolanrensen.docProcessor") version "{ VERSION }"
    ..
}

..

// Backup the kotlin source files location
val kotlinMainSources = kotlin.sourceSets.main.get().kotlin.sourceDirectories

// Create the processing task and point it to the right sources. 
// This can also be the test sources for instance.
val processKdocMain by creatingProcessDocTask(sources = kotlinMainSources) {

    // Optional. The target folder of the processed files. By default ${project.buildDir}/docProcessor/${taskName}.
    target = file(..)

    // The processors you want to use in this task.
    // The recommended order of default processors is as follows:
    processors = listOf(
        INCLUDE_DOC_PROCESSOR, // The @include processor
        INCLUDE_FILE_DOC_PROCESSOR, // The @includeFile processor
        ARG_DOC_PROCESSOR, // The @set and @get / $ processor
        COMMENT_DOC_PROCESSOR, // The @comment processor
        SAMPLE_DOC_PROCESSOR, // The @sample and @sampleNoComments processor
        EXPORT_AS_HTML_DOC_PROCESSOR, // The @exportAsHtmlStart and @exportAsHtmlEnd tags for @ExportAsHtml
        REMOVE_ESCAPE_CHARS_PROCESSOR, // The processor that removes escape characters

        "com.example.plugin.ExampleDocProcessor", // A custom processor if you have one, see below
    )

    // Optional. Send specific arguments to processors.
    arguments += ARG_DOC_PROCESSOR_LOG_NOT_FOUND to false

    // Optional dependencies for this task. These dependencies can introduce custom processors.
    dependencies {
        plugin("com.example:plugin:SOME_VERSION")
    }

    // Optional, defines where @ExportAsHtml will put the generated HTML files. By default ${project.buildDir}/docProcessor/${taskName}/htmlExports.
    exportAsHtml {
      dir = file("../docs/StardustDocs/snippets")
    }
}

// Modify all Jar tasks such that before running the Kotlin sources are set to 
// the target of processKdocMain and they are returned back to normal afterwards.
tasks.withType<Jar> {
    dependsOn(processKdocMain)
    outputs.upToDateWhen { false }

    doFirst {
        kotlin {
            sourceSets {
                main {
                    kotlin.setSrcDirs(processKdocMain.targets)
                }
            }
        }
    }

    doLast {
        kotlin {
            sourceSets {
                main {
                    kotlin.setSrcDirs(kotlinMainSources)
                }
            }
        }
    }
}

..

// As a bonus, this will update dokka to use the processed files as sources as well.
tasks.withType<org.jetbrains.dokka.gradle.AbstractDokkaLeafTask> {
    dokkaSourceSets {
        all {
            sourceRoot(processKdocMain.target.get())
        }
    }
}

build.gradle:

import nl.jolanrensen.docProcessor.gradle.*
import nl.jolanrensen.docProcessor.defaultProcessors.*
import org.gradle.jvm.tasks.Jar

..

plugins {
    id "nl.jolanrensen.docProcessor" version "{ VERSION }"
            ..
}

        ..

// Backup the kotlin source files location
def kotlinMainSources = kotlin.sourceSets.main.kotlin.sourceDirectories

// Create the processing task and point it to the right sources. 
// This can also be the test sources for instance.
def processKdocMain = tasks.register('processKdocMain', ProcessDocTask) {

    // Optional. The target folder of the processed files. By default ${project.buildDir}/docProcessor/${taskName}.
    target file(..)

    // The processors you want to use in this task.
    // The recommended order of default processors is as follows:
    processors(
            IncludeDocProcessorKt.INCLUDE_DOC_PROCESSOR, // The @include processor
            IncludeFileDocProcessorKt.INCLUDE_FILE_DOC_PROCESSOR, // The @includeFile processor
            ArgDocProcessorKt.ARG_DOC_PROCESSOR, // The @set and @get / $ processor
            CommentDocProcessorKt.COMMENT_DOC_PROCESSOR, // The @comment processor
            SampleDocProcessorKt.SAMPLE_DOC_PROCESSOR, // The @sample and @sampleNoComments processor
            ExportAsHtmlDocProcessorKt.EXPORT_AS_HTML_DOC_PROCESSOR, // The @exportAsHtmlStart and @exportAsHtmlEnd tags for @ExportAsHtml
            RemoveEscapeCharsProcessorKt.REMOVE_ESCAPE_CHARS_PROCESSOR, // The processor that removes escape characters

            "com.example.plugin.ExampleDocProcessor", // A custom processor if you have one, see below
    )

    // Optional. Send specific arguments to processors.
    arguments[IncludeArgDocProcessorKt.ARG_DOC_PROCESSOR] = false

    // Optional dependencies for this task. These dependencies can introduce custom processors.
    dependencies {
        plugin "com.example:plugin:SOME_VERSION"
    }

    // Optional, defines where @ExportAsHtml will put the generated HTML files. By default ${project.buildDir}/docProcessor/${taskName}/htmlExports.
    exportAsHtmlDir = file("../docs/StardustDocs/snippets")
    
}.get()

// Modify all Jar tasks such that before running the Kotlin sources are set to 
// the target of processKdocMain and they are returned back to normal afterwards.
tasks.withType(Jar).configureEach {
    dependsOn(processKdocMain)
    outputs.upToDateWhen { false }

    doFirst {
        kotlin {
            sourceSets {
                main {
                    kotlin.srcDirs = processKdocMain.targets
                }
            }
        }
    }

    doLast {
        kotlin {
            sourceSets {
                main {
                    kotlin.srcDirs = kotlinMainSources
                }
            }
        }
    }
}

        ..

// As a bonus, this will update dokka to use the processed files as sources as well.
        tasks.withType(org.jetbrains.dokka.gradle.AbstractDokkaLeafTask).configureEach {
            dokkaSourceSets.with {
                configureEach {
                    sourceRoot(processKdocMain.target.get())
                }
            }
        }

Recommended order of default processors

While you can use the processors in any order and leave out some or include others, the recommended order is as follows:

  • INCLUDE_DOC_PROCESSOR: The @include processor
  • INCLUDE_FILE_DOC_PROCESSOR: The @includeFile processor
  • ARG_DOC_PROCESSOR: The @set and @get / $ processor. This runs @set first and then @get / $.
  • COMMENT_DOC_PROCESSOR: The @comment processor
  • SAMPLE_DOC_PROCESSOR: The @sample and @sampleNoComments processor
  • EXPORT_AS_HTML_DOC_PROCESSOR: The @exportAsHtmlStart and @exportAsHtmlEnd tags for @ExportAsHtml
  • REMOVE_ESCAPE_CHARS_PROCESSOR: The processor that removes escape characters

This order ensures that @set/@get are processed after @include and @includeFile such that any arguments that appear by them are available for the @set/@get processor. The @comment processor is recommended to be after @set/@get too, as it can be used as a line break for tag blocks. @sample and @sampleNoComments are recommended to be last of the tag processors, as processing of inline tags inside comments of @sample might not be desired. Finally, the REMOVE_ESCAPE_CHARS_PROCESSOR is recommended to be last to clean up any escape characters that might have been introduced by the user to evade some parts of the docs from being processed.

Technicalities

Regarding block-tags KDocs and JavaDocs are structured in a tree-like structure and are thus also parsed and processed like that. For example, the following KDoc:

/**
 * Some extra text
 * @a [Test2]
 * Hi there!
 * @b source someFun
 * Some more text. {@c
 * @d [Test1] (
 * }
 * @e)
 */

will be split up in blocks as follows:

[
  "\nSome extra text",
  "@a [Test2]\nHi there!",
  "@b source someFun\nSome more text. (\n{@c [Test1] (\n}",
  "@e)\n",
]

This is also how tag processors receive their block-data (note that any newlines after the @tag are also included as part of the tag data).

Most tag processors only require a tiny number of arguments. They can decide what to do when they receive more arguments by the user. Most tag processors, like @include, @sample, and @includeFile all have systems in place that will preserve the content after the tag. Take this into account when writing your own processors.

To avoid any confusion, it's usually easier to stick to {@inline tags} as then it's clear which part of the doc belongs to the tag and what does not. Inline tags are processed before block tags per processor.

Take extra care when using tags that can introduce new tags, such as @include, as this will cause the structure of the doc to change mid-processing. Very powerful, but also potentially dangerous. If something weird happens, try to disable some processors to understand what's happening.

How it works

  • The sources provided to the plugin are read and analysed by Dokka's default SourceToDocumentableTranslators.
  • All Documentables are saved in a map by their path (e.g. com.example.plugin.Class1.function1) and their extension path.
  • Next, the documentation contents, location in the file, and indents are collected from each documentable in the map.
  • All processors are run in sequence on the collected documentables with their data:
    • All documentables are iterated over and tag replacement processors, like @include, will replace all tags with new content.
  • Finally, all files from the source are copied over to a destination folder and if there are any modifications that need to be made in a file, the specified ranges for each documentation are replaced with the new documentation.

Custom processors

You can create your plugin for the Gradle plugin with your own processors by either extending the abstract TagDocProcessor class or implementing the DocProcessor interface, depending on how much control you need over the docs.

Make sure to depend on the sources by adding the following to your build.gradle.kts or build.gradle file:

repositories {
    ..
    gradlePluginPortal()
}

dependencies {
    ..
    compileOnly("nl.jolanrensen.docProcessor:doc-processor-gradle-plugin:{ VERSION }")
}

Let's create a small example processor:

package com.example.plugin

class ExampleDocProcessor : TagDocProcessor() {

    /** We'll intercept @example tags. */
    override fun tagIsSupported(tag: String): Boolean =
        tag == "example"

    /** How `{@inner tags}` are processed. */
    override fun processInlineTagWithContent(
        tagWithContent: String,
        path: String,
        documentable: DocumentableWrapper,
    ): String = processContent(tagWithContent)

    /** How `  @normal tags` are processed. */
    override fun processBlockTagWithContent(
        tagWithContent: String,
        path: String,
        documentable: DocumentableWrapper,
    ): String = processContent(tagWithContent)

    // We can use the same function for both processInnerTagWithContent and processTagWithContent
    private fun processContent(tagWithContent: String): String {

        // We can log stuff, if we want to, using https://github.com/oshai/kotlin-logging
        logger.info { "Hi from the example logs!" }

        // We can get the content after the @example tag.
        val contentWithoutTag = tagWithContent
            .getTagArguments(tag = "example", numberOfArguments = 1)
            .single()
            .trimEnd() // remove trailing whitespaces/newlines

        // While we can play with the other arguments, let's just return some simple modified content
        var newContent =
            "Hi from the example doc processor! Here's the content after the @example tag: \"$contentWithoutTag\""

        // Since we trimmed all trailing newlines from the content, we'll add one back if they were there.
        if (tagWithContent.endsWith("\n"))
            newContent += "\n"

        return newContent
    }
}

For the processor to be detectable, we need to add it to the src/main/resources/META-INF/services/nl.jolanrensen.docProcessor.DocProcessor file:

com.example.plugin.ExampleDocProcessor

and then publish the project somewhere it can be used in other projects.

Add the published project as dependency in your other project's build.gradle.kts file in your created doc process task (as described in the How to Use section), both in the dependencies and in the processors list.

Now, if that project contains a function like:

/**
 * Main function.
 * @example Example
 */
fun main() {
    println("Hello World!")
}

The output will be:

/**
 * Main function.
 * Hi from the example doc processor! Here's the content after the @example tag: "Example"
 */
fun main() {
    println("Hello World!")
}

See the defaultProcessor folder in the project for more examples!

IntelliJ Plugin (Alpha)

Aside from a Gradle plugin, the project also contains an IntelliJ plugin that allows you to preview the rendered documentation directly in the IDE. image

Currently, the only way to try this is by building the plugin yourself from sources and installing it in IntelliJ. The plugin in its current state is unconfigurable and just uses the default processors as shown in the sample above. Also, it uses the IDE engine to resolve references. This is because it's a lot faster than my own engine + Dokka, but it does mean that there might be some differences with the preview and how it will look in the final docs. So, take this into account.

I'm still working on connecting it to the Gradle plugin somehow or provide a way to configure it correctly, but until then, you can use it as is and be more efficient in your documentation writing!