How to generate Java classes from Avro schemas as part of a Gradle build?

5.2k Views Asked by At

For Maven there is an official Avro plugin to generate Java classes from Avro schemas.

However, for Gradle there exists no official plugin.

There is davidmc24/gradle-avro-plugin, but it is no longer maintained and it is looking for a maintainer.

How can I generate Java classes from Avro schemas as part of a Gradle build?

2

There are 2 best solutions below

1
On BEST ANSWER

I have now created a simple Gradle task that generates the Avro Java classes.

import org.apache.avro.tool.SpecificCompilerTool

buildscript {
    dependencies {
        // Add the Avro code generation to the build dependencies so that it can be used in a Gradle task.
        classpath group: 'org.apache.avro', name: 'avro-tools', version: '1.11.1'
    }
}

plugins {
    // some project plugins
    id 'application'
}

def avroSchemasDir = "src/main/avro"
def avroCodeGenerationDir = "build/generated-main-avro-java"

// Add the generated Avro Java code to the Gradle source files.
sourceSets.main.java.srcDirs += [avroCodeGenerationDir]

dependencies {
    // some project dependencies
}

tasks.register('avroCodeGeneration') {
    // Define the task inputs and outputs for the Gradle up-to-date checks.
    inputs.dir(avroSchemasDir)
    outputs.dir(avroCodeGenerationDir)
    // The Avro code generation logs to the standard streams. Redirect the standard streams to the Gradle log.
    logging.captureStandardOutput(LogLevel.INFO);
    logging.captureStandardError(LogLevel.ERROR)
    doLast {
        // Run the Avro code generation.
        new SpecificCompilerTool().run(System.in, System.out, System.err, List.of(
                "-encoding", "UTF-8",
                "-string",
                "-fieldVisibility", "private",
                "-noSetters",
                "schema", "$projectDir/$avroSchemasDir".toString(), "$projectDir/$avroCodeGenerationDir".toString()
        ))
    }
}

tasks.withType(JavaCompile).configureEach {
    // Make Java compilation tasks depend on the Avro code generation task.
    dependsOn('avroCodeGeneration')
}

This task can be added to any Gradle build script.

0
On

The plugin project README was updated in December 2023 to state that the Apache Avro project is taking over the Gradle plugin.

This project is no longer maintained. Its code has been donated to the Apache Avro project, which will be handling releases going forward.

It is probably therefore reasonable to treat the plugin like an official plugin, even though as of this writing (December 2023) there hasn't been an Apache Avro release of the plugin.


To specifically answer the question about how to generate the Java classes from the Gradle build, the usage section of the plugin project README has details with an example:

Usage

Add the following to your build files. Substitute the desired version based on CHANGES.md.

settings.gradle:

pluginManagement {
    repositories {
        gradlePluginPortal()
        mavenCentral()
    }
}

build.gradle:

plugins {
    id "com.github.davidmc24.gradle.plugin.avro" version "VERSION"
}

Additionally, ensure that you have an implementation dependency on Avro, such as:

repositories {
    mavenCentral()
}
dependencies {
    implementation "org.apache.avro:avro:1.11.0"
}

If you now run gradle build, Java classes will be compiled from Avro files in src/main/avro. Actually, it will attempt to process an "avro" directory in every SourceSet (main, test, etc.)