Introduction

When working with Minecraft servers, adding additional third party libraries (like Javalin) can cause issues because of the custom class loaders of plugins or mods. This tutorial will provide a solution for switching class loaders and using Spigot’s dependency manager or Gradle Shadow’s dependency packaging plan.

The tutorial assumes you will create a basic Gradle and have some understanding of creating a Bukkit or BungeeCord plugin. If you need to learn more about this, you can click the following links: Gradle, Spigot, BungeeCord.

Someone else had provided the same solution before this tutorial, but it was not organized into a tutorial, this tutorial refers to their article (so thank them if you find this useful!).

Bukkit / Spigot / Paper

This tutorial will use Bukkit to refer to Bukkit, Spigot, and Paper in general, because Paper is actually a fork of Spigot, and Spigot is based on Bukkit, they are compatible with each other, and this tutorial is applicable to most other servers that fork Spigot or Paper.

BungeeCord / WaterFall

Using BungeeCord to refer to BungeeCord and WaterFall, BungeeCord’s solution is the same as Bukkit’s solution, just need to deal with their differences in dependencies and plugin API usage.

Dependencies

Add the necessary dependencies, Spigot API (which also includes Bukkit) or BungeeCord API, Javalin and two runtime dependencies for Javalin.

Add the statement to the Gradle configuration file build.gradle.

Bukkit / Spigot / Paper

repositories {
    mavenCentral()
    maven { url 'https://hub.spigotmc.org/nexus/content/repositories/snapshots/' }
}
dependencies {
    // https://hub.spigotmc.org/nexus/content/repositories/snapshots/
    compileOnly 'org.spigotmc:spigot-api:1.16.5-R0.1-SNAPSHOT'

    // https://mvnrepository.com/artifact/io.javalin/javalin
    implementation 'io.javalin:javalin:6.1.3'
    // https://mvnrepository.com/artifact/org.slf4j/slf4j-simple
    implementation 'org.slf4j:slf4j-simple:2.0.11'
    // https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-databind
    implementation 'com.fasterxml.jackson.core:jackson-databind:2.13.2'
}

BungeeCord / WaterFall

repositories {
    mavenCentral()
    maven { url 'https://oss.sonatype.org/content/repositories/snapshots' }
}
dependencies {
    compileOnly 'net.md-5:bungeecord-api:1.16-R0.5-SNAPSHOT'

    // https://mvnrepository.com/artifact/io.javalin/javalin
    implementation 'io.javalin:javalin:6.1.3'
    // https://mvnrepository.com/artifact/org.slf4j/slf4j-simple
    implementation 'org.slf4j:slf4j-simple:2.0.11'
    // https://mvnrepository.com/artifact/com.fasterxml.jackson.core/jackson-databind
    implementation 'com.fasterxml.jackson.core:jackson-databind:2.13.2'
}

Plugin Main Class

As described earlier, Bukkit uses custom class loaders that interfere with Javalin-dependent class loading, causing exceptions like as java.lang.NoClassDefFoundError and java.lang.ClassNotFoundException to be thrown.

As shown in the following code, you need to temporarily switch the class loader to the main class loader of the current plugin, and then switch to the default class loader after Javalin is instantiated. At this time, the Javalin instance can already be in the context of the default class loader used in.

// org.bukkit.plugin.java.JavaPlugin is the plugin interface of Bukkit,
// BungeeCord should be changed to net.md_5.bungee.api.plugin.Plugin.
// BungeeCord: public class JavalinPlugin extends Plugin
public class JavalinPlugin extends JavaPlugin {
    @Override
    public void onEnable() {
        // Temporarily switch the plugin classloader to load Javalin.
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        // BungeeCord:  Thread.currentThread().setContextClassLoader(this.getClass().getClassLoader());
        Thread.currentThread().setContextClassLoader(this.getClassLoader());
        // Create a Javalin instance.
        Javalin app = Javalin.create().start(8080);
        // Restore default loader.
        Thread.currentThread().setContextClassLoader(classLoader);
        // The created instance can be used outside the class loader.
        app.get("/", ctx -> ctx.result("Hello World!"));
        // log
        getLogger().info("JavalinPlugin is enabled");
    }

    @Override
    public void onDisable() {
        getLogger().info("JavalinPlugin is disabled");
    }
}

Note your plugin interface, Bukkit and BungeeCord are slightly different, already marked in the comments.

Thread.currentThread().getContextClassLoader() Default class loader for getting context, keep it for switching after load.

Thread.currentThread().setContextClassLoader(this.getClassLoader()) is to switch the class loader to the class loader of the current plugin. In other classes, you can use YourMainClass.class.getClassLoader() or YourMainClass.getClass().getClassLoader() replace this.getClassLoader()

After instantiating Javalin, use setContextClassLoader(classLoader) to reset the context class loader to the default class loader.

Wait, don’t go! One more step to finish! :D

Use Gradle Shadow for packaging.

You also need to package Javalin into the plugin and handle possible dependency conflicts, this method works for all versions of Bukkit and BungeeCord.

Add the statement to the Gradle configuration file build.gradle.

plugins {
    id 'java'
    id 'com.github.johnrengelman.shadow' version '7.1.2'
}

Relocation in the shadowJar configuration node is to prevent conflicts with the same dependencies carried by other plugins. This step is not required, but is recommended if your plugin plans to release it into the public domain to prevent incompatibilities.

A common practice is to relocation the entire dependent library.

shadowJar {
    relocate 'io.javalin:javalin:6.1.3', 'shadow.io.javalin'
    relocate 'org.slf4j:slf4j-simple:2.0.11', 'shadow.org.slf4j'
    relocate 'com.fasterxml.jackson.core:jackson-databind:2.13.2', 'shadow.com.fasterxml.jackson.core'
}

For more information on the Gradle Shadow plugin see https://imperceptiblethoughts.com/shadow/

Using the plugin dependency manager

This method only works with Spigot / Paper plugins higher than 1.17.

No need to change build.gradle , just add in plugin description file plugin.yml .

libraries:
  - "io.javalin:javalin:6.1.3"
  - "org.slf4j:slf4j-simple:2.0.11"
  - "com.fasterxml.jackson.core:jackson-databind:2.13.2"

It has the same format as the Gradle dependency url short format, like “Group Id:Artifact Id:Version”. It should be noted that only the dependencies of the Maven central repository are supported.

Build

Run Gradle tasks with Gradle Shadow: ./gradlew shadowJar
Run Gradle tasks with plugin dependency manager: ./gradlew build