LuckPerms has a complete developer API, which allows other plugins on the server to read and modify LuckPerms data, and easily integrate LuckPerms deeply into existing plugins and systems.
The API uses Semantic Versioning, meaning whenever a non-backwards compatible change is made, the major version will increment. You can rest assured knowing your integration will not break between versions, providing the major version remains the same.
The current API release is 5.0
.
- The API package in LuckPerms is
net.luckperms.api
. - JavaDocs are available either in a standard JavaDoc layout, or within the API source code.
- Version
2.x
represented the initial release of the API. - Version
3.x
(19th Feb 17) introduced a number of backwards incompatible changes. [changelog] - Version
4.x
(7th Nov 17) introduced a number of backwards incompatible changes. [changelog] - Version
5.0
was a complete rewrite of the API. Bridging tools are provided to maintain compatibility with older versions.
The API artifact is published to the Maven Central repository.
If you're using Maven, simply add this to the dependencies
section of your POM.
<dependencies>
<dependency>
<groupId>net.luckperms</groupId>
<artifactId>api</artifactId>
<version>5.0</version>
<scope>provided</scope>
</dependency>
</dependencies>
If you're using Gradle, you need to add these lines to your build script.
repositories {
mavenCentral()
}
dependencies {
compile 'net.luckperms:api:5.0'
}
Note:
compile
can be optionally replaced by eitherapi
orimplementation
on newer Gradle versions to distinguish whether dependant projects should transitively pick up the LuckPerms API dependency.
If you want to manually add the API dependency to your classpath, you can obtain the jar by:
- Navigating to
https://repo1.maven.org/maven2/net/luckperms/api/
- Selecting the version you wish to use
- Downloading the jar titled
api-x.x.jar
The root API interface is LuckPerms
. You need to obtain an instance of this interface in order to do anything.
It can be obtained in a number of ways.
When the plugin is enabled, an instance of LuckPerms
will be provided in the Bukkit ServicesManager. (obviously you need to be writing your plugin for Bukkit!)
RegisteredServiceProvider<LuckPerms> provider = Bukkit.getServicesManager().getRegistration(LuckPerms.class);
if (provider != null) {
LuckPerms api = provider.getProvider();
}
When the plugin is enabled, an instance of LuckPerms
will be provided in the Sponge ServicesManager. (obviously you need to be writing your plugin for Sponge!)
Optional<ProviderRegistration<LuckPerms>> provider = Sponge.getServiceManager().getRegistration(LuckPerms.class);
if (provider.isPresent()) {
LuckPerms api = provider.get().getProvider();
}
When the plugin is enabled, an instance of LuckPerms
can be obtained statically from the LuckPermsProvider
class. (this will work on all platforms)
Note: this method will throw an IllegalStateException
if the API is not loaded.
LuckPerms api = LuckPermsProvider.get();
Now you've added the API classes to your project, and obtained an instance of the LuckPerms
, you're almost ready to start using the API. However, before you go any further, please make sure you read and understand the information below.
- All LuckPerms internals are thread-safe. You can safely interact with the API from scheduler tasks (or just generally from other threads)
- This also extends to the permission querying methods in Bukkit/Bungee/Sponge. These can be safely called async when LuckPerms is being used as the permissions plugin.
- In cases where methods return classes from the Java collections framework, assume that the returned methods are always immutable, unless indicated otherwise. (in the JavaDocs)
- This means that you cannot make changes to any returned collections.
- Some methods are not "main thread friendly", meaning if they are called from the main Minecraft Server thread, the server will lag.
- This is because many methods conduct I/O with either the file system or the network.
- In most cases, these methods return CompletableFutures.
- Futures can be an initially complex paradigm for some users - however, it is crucial that you have at least a basic understanding of how they work before attempting to use them.
- As a general rule, it is advised that if it's convenient to do so, you conduct as much work with the API as possible within async scheduler tasks. Some methods don't return futures, but may still involve a number of relatively complex computations.
This is a super quick guide. If you'd like more comprehensive info, see the CompletableFuture or CompletionStage JavaDoc pages.
For the purposes of explaining, take the following method in the ActionLogger
class.
CompletableFuture<ActionLog> getLog();
After calling the method, we get a CompletableFuture<Log>
- the object we actually want is the ActionLog
. The CompletableFuture
represents the result of some computation (in this case the computation to obtain the ActionLog), and provides us with methods to obtain the ActionLog
object.
If the context of our method call is already asynchronous (if we're calling the method from an async scheduler task), then we can do-away with the future entirely.
/*
Calling this method effectively "requests" an ActionLog from the API.
However, it's unlikely that the log will be available immediately...
We need to wait for it to be supplied.
*/
CompletableFuture<ActionLog> logFuture = actionLogger.getLog();
/*
Since we're already on an async thread, it doesn't matter how long we
have to wait for the elusive Log to show up.
The #join method will block - and wait until the Log has been supplied,
and then return it.
If for whatever reason the process to obtain a ActionLog threw an exception,
this method will rethrow an the same exception wrapped in a CompletionException
*/
ActionLog log = logFuture.join();
An alternative to using #join
is to register a callback with the Future, to be executed once the Log
is supplied.
If we need to use the instance on the main server thread, then a special executor can be passed to the callback is executed on the server thread.
// The executor to run the callback on
Executor executor = runnable -> Bukkit.getScheduler().runTask(this, runnable);
// To be called once the Log is obtained.
logFuture.whenCompleteAsync(new BiConsumer<ActionLog, Throwable>() { // can be reduced to a lambda, I've left it as an anonymous class for clarity
@Override
public void accept(ActionLog log, Throwable exception) {
if (exception != null) {
// There was some error whilst getting the log.
return;
}
// Use the log for something...
}
}, executor);
If you don't care about errors, this can be simplified further.
logFuture.thenAcceptAsync(log -> { /* Use the log for something */ }, executor);
The CompletableFuture class can initially be very confusing to use (it's still a relatively new API in Java!), however it is a great way to encapsulate async computations, and in the case of Minecraft, ensures that users don't accidently block the server thread waiting on lengthy I/O calls.
- The vast majority of LuckPerms' work is done in async tasks away from the server thread.
- With that in mind, it would be silly to call LuckPerms events synchronously - meaning that, without exception, all events listeners are called asynchronously.
Please keep in mind that many parts of Bukkit, Sponge and the Minecraft server in general are not thread-safe, and should only be interacted with from the server thread. If you need to use Bukkit or Sponge methods from within LuckPerms event listeners or callbacks, you need to perform your action using the scheduler.