Zapper is a powerful runtime dependency downloader for Bukkit that aims to abstract away all difficulties that arise with such task.
- 🗺️ Relocation: Relocate dependencies to avoid conflicts with other plugins or Bukkit's own dependencies
- 🚌 Transitive dependencies: Dependencies of dependencies will also be downloaded to ensure no classes are missing at runtime
- 🔥 Does not require
org.bukkit.plugin.Plugin
instances: This is a very significant convenience of Zapper, as it allows you to:- Not have to worry about loading dependencies before or after your plugin loads.
- Your main class can use dependency classes freely (e.g. download Kotlin and your main class would be in Kotlin)
- Dependencies are loaded before your plugin loads
- 🐘 Seamless integration with Gradle and ShadowJar
- 🍵 Works on all Java versions from 8 to 22
Zapper provides a Gradle plugin that seamlessly integrates with your build script. To apply it:
build.gradle:
plugins {
id 'com.github.johnrengelman.shadow' version '8.1.1'
id 'io.github.revxrsal.zapper' version '1.0.3'
}
build.gradle.kts:
plugins {
id("com.github.johnrengelman.shadow") version "8.1.1"
id("io.github.revxrsal.zapper") version "1.0.3"
}
Then, add your dependencies using the zap
configuration block:
dependencies {
// an example dependency
zap("com.squareup.moshi:moshi:1.15.2")
// you can use the dependency notation and exclude modules as you
// would with other configurations
zap("com.squareup.moshi", "moshi", "1.15.2") {
exclude(module: "kotlin")
}
zap("org.jetbrains.kotlin:kotlin-stdlib-jdk8:2.1.0")
}
You can provide additional configuration using the zapper
extension:
zapper {
// directory to download dependencies in
libsFolder = "libraries"
// the prefix for relocating libraries
relocationPrefix = "myplugin.libs"
// repositories to fetch dependencies from
//
// by default: includes maven central
repositories {
// example repository
maven("https://jitpack.io/")
// optional: use all repositories declared in this
// file if you don't want to re-include everything here
includeProjectRepositories()
}
// relocate libraries here
// com.squareup.moshi --> myplugin.libs.moshi
relocate("com.squareup.moshi", "moshi")
}
Then, in your plugin, extend ZapperJavaPlugin
instead of JavaPlugin
:
public final class MyPlugin extends ZapperJavaPlugin {
// use your dependencies here! don't think twice :D
private final Moshi moshi;
}
Finally, build with shadowJar
:
./gradlew shadowJar
Sadly, Maven usage is not as seamless as the Gradle one. However, you can use the API provided by Zapper to download dependencies at runtime:
To add the Zapper API:
<dependency>
<groupId>io.github.revxrsal</groupId>
<artifactId>zapper.api</artifactId>
<version>1.0.3</version>
<scope>compile</scope>
</dependency>
See Zapper dependency API usage
- Create a base class that declares your dependencies. Don't include your actual plugin logic here. Your main plugin class should extend this base class.
public abstract class MyPluginBase extends JavaPlugin {
static {
File libraries = new File(
ClassLoaderReader.getDataFolder(MyPluginBase.class),
"libraries" // libraries folder
);
if (!libraries.exists()) {
PluginDescriptionFile pdf = ClassLoaderReader.getDescription(ZapperJavaPlugin.class);
Bukkit.getLogger().info("[" + pdf.getName() + "] It appears you're running " + pdf.getName() + " for the first time.");
Bukkit.getLogger().info("[" + pdf.getName() + "] Please give me a few seconds to install dependencies. This is a one-time process.");
}
DependencyManager dependencyManager = new DependencyManager(
libraries,
URLClassLoaderWrapper.wrap((URLClassLoader) MyPluginBase.class.getClassLoader())
);
// add your repositories
dependencyManager.repository(Repository.mavenCentral());
dependencyManager.repository(Repository.maven("https://jitpack.io"));
// add your dependencies
dependencyManager.dependency("com.squareup.moshi:moshi:1.15.2");
// IMPORTANT NOTE: Beware that this path may get relocated/changed
// by your build tool!!! Escape it using runtime tricks if necessary
dependencyManager.relocate(new Relocation(
"com{}squareup{}moshi".replace("{}", "."),
"myplugin.moshi"
));
dependencyManager.load();
}
}
- Create your actual main class and extend the base class you have just defined:
public final class MyPlugin extends MyPluginBase {
// your use dependencies here as you like
}
If you, for some reason, would like to extend a different class than a JavaPlugin, and are using the Gradle plugin, you can avoid extending the ZapperJavaPlugin class by doing the following:
- Declare a base class that extends the custom JavaPlugin class:
public abstract class MyPluginBase extends CustomJavaPlugin {
}
- Add the following static block to
MyPluginBase
:
static {
RuntimeLibPluginConfiguration config = RuntimeLibPluginConfiguration.parse();
File libraries = new File(ClassLoaderReader.getDataFolder(MyPluginBase.class), config.getLibsFolder());
if (!libraries.exists()) {
PluginDescriptionFile pdf = ClassLoaderReader.getDescription(MyPluginBase.class);
Bukkit.getLogger().info("[" + pdf.getName() + "] It appears you're running " + pdf.getName() + " for the first time.");
Bukkit.getLogger().info("[" + pdf.getName() + "] Please give me a few seconds to install dependencies. This is a one-time process.");
}
DependencyManager dependencyManager = new DependencyManager(
libraries,
URLClassLoaderWrapper.wrap((URLClassLoader) MyPluginBase.class.getClassLoader())
);
config.getDependencies().forEach(dependencyManager::dependency);
config.getRepositories().forEach(dependencyManager::repository);
config.getRelocations().forEach(dependencyManager::relocate);
dependencyManager.load();
}
- Extend
MyPluginBase
public final class MyPlugin extends MyPluginBase {
// use your dependencies here as you like!
}
If Zapper has made your life significantly easier or you're feeling particularly generous, consider sponsoring the project! It's a great way to support the many hours I've spent maintaining this library and keeps me motivated. Please don't sponsor if you can't afford it.
Huge thanks to those who donated! 😄