From 20f24a2e9206dae2dda43636b0ccfbb8dff8728e Mon Sep 17 00:00:00 2001 From: Napster Date: Fri, 27 Mar 2020 20:41:09 +0100 Subject: [PATCH 01/56] Pin TLSv1.2 --- .../src/main/java/lavalink/server/io/SocketContext.kt | 10 +++++++++- build.gradle | 2 +- 2 files changed, 10 insertions(+), 2 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt index a8cd27326..86d68d7ca 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt @@ -33,6 +33,8 @@ import org.json.JSONObject import org.slf4j.LoggerFactory import org.springframework.web.socket.WebSocketSession import org.springframework.web.socket.adapter.standard.StandardWebSocketSession +import org.xnio.OptionMap +import org.xnio.Options import space.npstr.magma.MagmaFactory import space.npstr.magma.api.MagmaApi import space.npstr.magma.api.event.MagmaEvent @@ -56,9 +58,15 @@ class SocketContext internal constructor( private val log = LoggerFactory.getLogger(SocketContext::class.java) } - internal val magma: MagmaApi = MagmaFactory.of { socketServer.getAudioSendFactory(it) } + internal val magma: MagmaApi = MagmaFactory.of( + { socketServer.getAudioSendFactory(it) }, + OptionMap.EMPTY, + OptionMap.create(Options.SSL_PROTOCOL, "TLSv1.2") + ) + //guildId <-> Player val players = ConcurrentHashMap() + @Volatile var sessionPaused = false private val resumeEventQueue = ConcurrentLinkedQueue() diff --git a/build.gradle b/build.gradle index 78450736b..f3fd98c24 100644 --- a/build.gradle +++ b/build.gradle @@ -54,7 +54,7 @@ subprojects { //@formatter:off lavaplayerVersion = '1.3.38' lavaplayerIpRotatorVersion = '0.1.7' - magmaVersion = '0.12.5' + magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' From 24d249834ebe178b7d0e7c7ee3bfccd9465f1ad4 Mon Sep 17 00:00:00 2001 From: thewsomeguy Date: Sun, 29 Mar 2020 19:41:15 -0500 Subject: [PATCH 02/56] Updated lavaplayer & IP rotator --- .../lavalink/server/config/AudioPlayerConfiguration.kt | 8 ++++---- build.gradle | 4 ++-- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/config/AudioPlayerConfiguration.kt b/LavalinkServer/src/main/java/lavalink/server/config/AudioPlayerConfiguration.kt index 6f5fe73ab..953fb1a2b 100644 --- a/LavalinkServer/src/main/java/lavalink/server/config/AudioPlayerConfiguration.kt +++ b/LavalinkServer/src/main/java/lavalink/server/config/AudioPlayerConfiguration.kt @@ -14,7 +14,7 @@ import com.sedmelluq.discord.lavaplayer.source.soundcloud.SoundCloudAudioSourceM import com.sedmelluq.discord.lavaplayer.source.twitch.TwitchStreamAudioSourceManager import com.sedmelluq.discord.lavaplayer.source.vimeo.VimeoAudioSourceManager import com.sedmelluq.discord.lavaplayer.source.youtube.YoutubeAudioSourceManager -import com.sedmelluq.lava.extensions.youtuberotator.YoutubeIpRotator +import com.sedmelluq.lava.extensions.youtuberotator.YoutubeIpRotatorSetup import com.sedmelluq.lava.extensions.youtuberotator.planner.AbstractRoutePlanner import com.sedmelluq.lava.extensions.youtuberotator.planner.BalancingIpRoutePlanner import com.sedmelluq.lava.extensions.youtuberotator.planner.NanoIpRoutePlanner @@ -50,9 +50,9 @@ class AudioPlayerConfiguration { if (routePlanner != null) { val retryLimit = serverConfig.ratelimit?.retryLimit ?: -1 when { - retryLimit < 0 -> YoutubeIpRotator.setup(youtube, routePlanner) - retryLimit == 0 -> YoutubeIpRotator.setup(youtube, routePlanner, Int.MAX_VALUE) - else -> YoutubeIpRotator.setup(youtube, routePlanner, retryLimit) + retryLimit < 0 -> YoutubeIpRotatorSetup(routePlanner).forSource(youtube).setup() + retryLimit == 0 -> YoutubeIpRotatorSetup(routePlanner).forSource(youtube).withRetryLimit(Int.MAX_VALUE).setup() + else -> YoutubeIpRotatorSetup(routePlanner).forSource(youtube).withRetryLimit(retryLimit).setup() } } diff --git a/build.gradle b/build.gradle index f3fd98c24..f70892104 100644 --- a/build.gradle +++ b/build.gradle @@ -52,8 +52,8 @@ subprojects { ext { //@formatter:off - lavaplayerVersion = '1.3.38' - lavaplayerIpRotatorVersion = '0.1.7' + lavaplayerVersion = '1.3.40' + lavaplayerIpRotatorVersion = '0.2.3' magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' From d1f1c53c7dd364b93bec2bb63c1c8d4c2dcb3630 Mon Sep 17 00:00:00 2001 From: Devoxin <15076404+Devoxin@users.noreply.github.com> Date: Sun, 5 Apr 2020 00:46:11 +0100 Subject: [PATCH 03/56] Bump Lavaplayer dependency --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index f3fd98c24..135d9ff5f 100644 --- a/build.gradle +++ b/build.gradle @@ -52,7 +52,7 @@ subprojects { ext { //@formatter:off - lavaplayerVersion = '1.3.38' + lavaplayerVersion = '1.3.44' lavaplayerIpRotatorVersion = '0.1.7' magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' From 61bbe0ce29aa2d6395cdbb1db482ee0757822ffb Mon Sep 17 00:00:00 2001 From: Devoxin <15076404+Devoxin@users.noreply.github.com> Date: Tue, 7 Apr 2020 01:14:03 +0100 Subject: [PATCH 04/56] Update build.gradle --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index 135d9ff5f..a8be90ed6 100644 --- a/build.gradle +++ b/build.gradle @@ -52,7 +52,7 @@ subprojects { ext { //@formatter:off - lavaplayerVersion = '1.3.44' + lavaplayerVersion = '1.3.46' lavaplayerIpRotatorVersion = '0.1.7' magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' From aed1559d18dce6aee437d3337cca43c2615acf86 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Thu, 9 Apr 2020 12:51:54 +0200 Subject: [PATCH 05/56] Add koe to the classpath --- LavalinkServer/build.gradle | 1 + build.gradle | 1 + 2 files changed, 2 insertions(+) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index bf641caf4..ca5ac2071 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -34,6 +34,7 @@ bootRun { } dependencies { + compile group: 'moe.kyokobot', name: 'koe', version: koeVersion compile group: 'space.npstr.Magma', name: 'magma', version: magmaVersion compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion diff --git a/build.gradle b/build.gradle index a8be90ed6..b07980375 100644 --- a/build.gradle +++ b/build.gradle @@ -54,6 +54,7 @@ subprojects { //@formatter:off lavaplayerVersion = '1.3.46' lavaplayerIpRotatorVersion = '0.1.7' + koeVersion = '0.2.0' magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' From 83cb9b53c7d796046d76ace6d08206b9af95b114 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Thu, 9 Apr 2020 12:52:23 +0200 Subject: [PATCH 06/56] Fix redundancy warning --- .../src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt b/LavalinkServer/src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt index b86ac1998..ac6e10f10 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt @@ -65,7 +65,7 @@ class RoutePlannerRestHandler(private val routePlanner: AbstractRoutePlanner?) { */ private fun getDetailBlock(planner: AbstractRoutePlanner): IRoutePlannerStatus { val ipBlock = planner.ipBlock - val ipBlockStatus = IpBlockStatus(ipBlock.type.simpleName, ipBlock.size.toString() ?: "0") + val ipBlockStatus = IpBlockStatus(ipBlock.type.simpleName, ipBlock.size.toString()) val failingAddresses = planner.failingAddresses val failingAddressesStatus = failingAddresses.entries.map { From ffbd3010ea55863d83c7aa870ed199a167d1d08b Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Fri, 10 Apr 2020 14:08:41 +0200 Subject: [PATCH 07/56] Use Koe instead of Magma --- LavalinkServer/build.gradle | 1 + .../server/config/KoeConfiguration.kt | 33 +++++++++ .../java/lavalink/server/io/SocketContext.kt | 74 +++++++++++-------- .../java/lavalink/server/io/SocketServer.kt | 33 +++------ .../lavalink/server/io/WebSocketHandlers.kt | 41 +++------- .../java/lavalink/server/player/Player.java | 18 ++--- build.gradle | 2 +- 7 files changed, 108 insertions(+), 94 deletions(-) create mode 100644 LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index ca5ac2071..a90ee05b0 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -35,6 +35,7 @@ bootRun { dependencies { compile group: 'moe.kyokobot', name: 'koe', version: koeVersion + compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion compile group: 'space.npstr.Magma', name: 'magma', version: magmaVersion compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion diff --git a/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt b/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt new file mode 100644 index 000000000..d9cf73793 --- /dev/null +++ b/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt @@ -0,0 +1,33 @@ +package lavalink.server.config + +import moe.kyokobot.koe.KoeOptions +import moe.kyokobot.koe.codec.udpqueue.UdpQueueFramePollerFactory +import org.slf4j.Logger +import org.slf4j.LoggerFactory +import org.springframework.context.annotation.Bean +import org.springframework.context.annotation.Configuration + +@Configuration +class KoeConfiguration(val serverConfig: ServerConfig) { + + private val log: Logger = LoggerFactory.getLogger(KoeConfiguration::class.java) + + @Bean + fun koeOptions(): KoeOptions = KoeOptions.builder().apply { + log.info("OS: " + System.getProperty("os.name") + ", Arch: " + System.getProperty("os.arch")) + val os = System.getProperty("os.name") + val arch = System.getProperty("os.arch") + + // Maybe add Windows natives back? + val nasSupported = os.contains("linux", ignoreCase = true) + && arch.equals("amd64", ignoreCase = true) + + if (nasSupported) { + log.info("Enabling JDA-NAS") + setFramePollerFactory(UdpQueueFramePollerFactory()) + } else { + log.warn("This system and architecture appears to not support native audio sending! " + + "GC pauses may cause your bot to stutter during playback.") + } + }.create() +} \ No newline at end of file diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt index 86d68d7ca..0ddcde51c 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt @@ -23,47 +23,41 @@ package lavalink.server.io import com.sedmelluq.discord.lavaplayer.player.AudioPlayerManager -import lavalink.server.player.Player -import space.npstr.magma.api.MagmaMember import io.undertow.websockets.core.WebSocketCallback import io.undertow.websockets.core.WebSocketChannel import io.undertow.websockets.core.WebSockets import io.undertow.websockets.jsr.UndertowSession +import lavalink.server.player.Player +import moe.kyokobot.koe.KoeClient +import moe.kyokobot.koe.KoeEventAdapter +import moe.kyokobot.koe.VoiceConnection +import moe.kyokobot.koe.VoiceServerInfo import org.json.JSONObject import org.slf4j.LoggerFactory import org.springframework.web.socket.WebSocketSession import org.springframework.web.socket.adapter.standard.StandardWebSocketSession -import org.xnio.OptionMap -import org.xnio.Options -import space.npstr.magma.MagmaFactory -import space.npstr.magma.api.MagmaApi import space.npstr.magma.api.event.MagmaEvent import space.npstr.magma.api.event.WebSocketClosed import java.util.* -import java.util.concurrent.* import java.util.concurrent.ConcurrentHashMap +import java.util.concurrent.ConcurrentLinkedQueue import java.util.concurrent.Executors import java.util.concurrent.ScheduledExecutorService +import java.util.concurrent.ScheduledFuture import java.util.concurrent.TimeUnit -import java.util.function.Supplier class SocketContext internal constructor( val audioPlayerManager: AudioPlayerManager, var session: WebSocketSession, private val socketServer: SocketServer, - val userId: String + val userId: String, + private val koe: KoeClient ) { companion object { private val log = LoggerFactory.getLogger(SocketContext::class.java) } - internal val magma: MagmaApi = MagmaFactory.of( - { socketServer.getAudioSendFactory(it) }, - OptionMap.EMPTY, - OptionMap.create(Options.SSL_PROTOCOL, "TLSv1.2") - ) - //guildId <-> Player val players = ConcurrentHashMap() @@ -75,7 +69,7 @@ class SocketContext internal constructor( var resumeKey: String? = null var resumeTimeout = 60L // Seconds private var sessionTimeoutFuture: ScheduledFuture? = null - private val executor: ScheduledExecutorService + private val executor: ScheduledExecutorService = Executors.newSingleThreadScheduledExecutor() val playerUpdateService: ScheduledExecutorService val playingPlayers: List @@ -87,9 +81,6 @@ class SocketContext internal constructor( init { - magma.eventStream.subscribe { this.handleMagmaEvent(it) } - - executor = Executors.newSingleThreadScheduledExecutor() executor.scheduleAtFixedRate(StatsTask(this, socketServer), 0, 1, TimeUnit.MINUTES) playerUpdateService = Executors.newScheduledThreadPool(2) { r -> @@ -122,6 +113,26 @@ class SocketContext internal constructor( } } + /** + * Gets or creates a voice connection + */ + fun getVoiceConnection(guild: Long): VoiceConnection { + var conn = koe.getConnection(guild) + if (conn == null) { + conn = koe.createConnection(guild) + conn.registerListener(EventHandler(guild.toString())) + } + return conn + } + + /** + * Disposes of a voice connection + */ + fun destroy(guild: Long) { + players.remove(guild.toString())?.stop() + koe.destroyConnection(guild) + } + fun pause() { sessionPaused = true sessionTimeoutFuture = executor.schedule({ @@ -177,16 +188,21 @@ class SocketContext internal constructor( log.info("Shutting down " + playingPlayers.size + " playing players.") executor.shutdown() playerUpdateService.shutdown() - players.keys.forEach { guildId -> - val member = MagmaMember.builder() - .userId(userId) - .guildId(guildId) - .build() - magma.removeSendHandler(member) - magma.closeConnection(member) - } - players.values.forEach(Player::stop) - magma.shutdown() + koe.close() + } + + private inner class EventHandler(private val guildId: String) : KoeEventAdapter() { + override fun gatewayClosed(code: Int, reason: String?, byRemote: Boolean) { + val out = JSONObject() + out.put("op", "event") + out.put("type", "WebSocketClosedEvent") + out.put("guildId", guildId) + out.put("reason", reason) + out.put("code", code) + out.put("byRemote", byRemote) + + send(out) + } } } diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt index d16cfd51f..d8732cff5 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt @@ -29,6 +29,8 @@ import lavalink.server.config.AudioSendFactoryConfiguration import lavalink.server.config.ServerConfig import lavalink.server.player.Player import lavalink.server.util.Util +import moe.kyokobot.koe.Koe +import moe.kyokobot.koe.KoeOptions import net.dv8tion.jda.api.audio.factory.IAudioSendFactory import org.json.JSONObject import org.slf4j.LoggerFactory @@ -40,13 +42,13 @@ import org.springframework.web.socket.handler.TextWebSocketHandler import space.npstr.magma.api.Member import java.util.* import java.util.concurrent.ConcurrentHashMap -import java.util.function.Supplier @Service class SocketServer( private val serverConfig: ServerConfig, private val audioPlayerManager: AudioPlayerManager, - private val audioSendFactoryConfiguration: AudioSendFactoryConfiguration + private val audioSendFactoryConfiguration: AudioSendFactoryConfiguration, + koeOptions: KoeOptions ) : TextWebSocketHandler() { // userId <-> shardCount @@ -56,6 +58,7 @@ class SocketServer( @Suppress("LeakingThis") private val handlers = WebSocketHandlers(contextMap) private val resumableSessions = mutableMapOf() + private val koe = Koe.koe(koeOptions) companion object { private val log = LoggerFactory.getLogger(SocketServer::class.java) @@ -92,7 +95,13 @@ class SocketServer( shardCounts[userId] = shardCount - contextMap[session.id] = SocketContext(audioPlayerManager, session, this, userId) + contextMap[session.id] = SocketContext( + audioPlayerManager, + session, + this, + userId, + koe.newClient(userId.toLong()) + ) log.info("Connection successfully established from " + session.remoteAddress!!) } @@ -157,24 +166,6 @@ class SocketServer( } } - fun getAudioSendFactory(member: Member): IAudioSendFactory { - val shardCount = shardCounts.getOrDefault(member.userId, 1) - val shardId = Util.getShardFromSnowflake(member.guildId, shardCount) - - return sendFactories.computeIfAbsent(shardId % audioSendFactoryConfiguration.audioSendFactoryCount - ) { - val customBuffer = serverConfig.bufferDurationMs - val nativeAudioSendFactory: NativeAudioSendFactory - nativeAudioSendFactory = if (customBuffer != null) { - NativeAudioSendFactory(customBuffer) - } else { - NativeAudioSendFactory() - } - - AsyncPacketProviderFactory.adapt(nativeAudioSendFactory) - } - } - internal fun onSessionResumeTimeout(context: SocketContext) { resumableSessions.remove(context.resumeKey) context.shutdown() diff --git a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt index 31222ef65..95a89f653 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt @@ -1,6 +1,7 @@ package lavalink.server.io import lavalink.server.util.Util +import moe.kyokobot.koe.VoiceServerInfo import org.json.JSONObject import org.slf4j.Logger import org.slf4j.LoggerFactory @@ -16,28 +17,17 @@ class WebSocketHandlers(private val contextMap: Map) { fun voiceUpdate(session: WebSocketSession, json: JSONObject) { val sessionId = json.getString("sessionId") - val guildId = json.getString("guildId") + val guildId = json.getLong("guildId") val event = json.getJSONObject("event") - val endpoint = event.optString("endpoint") - val token = event.getString("token") + val endpoint: String? = event.optString("endpoint") + val token: String = event.getString("token") //discord sometimes send a partial server update missing the endpoint, which can be ignored. - if (endpoint == null || endpoint.isEmpty()) { - return - } + endpoint ?: return val sktContext = contextMap[session.id]!! - val member = MagmaMember.builder() - .userId(sktContext.userId) - .guildId(guildId) - .build() - val serverUpdate = MagmaServerUpdate.builder() - .sessionId(sessionId) - .endpoint(endpoint) - .token(token) - .build() - sktContext.magma.provideVoiceServerUpdate(member, serverUpdate) + sktContext.getVoiceConnection(guildId).connect(VoiceServerInfo(sessionId, endpoint, token)); } fun play(session: WebSocketSession, json: JSONObject) { @@ -65,13 +55,8 @@ class WebSocketHandlers(private val contextMap: Map) { val context = contextMap[session.id]!! - val m = MagmaMember.builder() - .userId(context.userId) - .guildId(json.getString("guildId")) - .build() - context.magma.setSendHandler(m, context.getPlayer(json.getString("guildId"))) - - SocketServer.sendPlayerUpdate(ctx, player) + context.getVoiceConnection(player.guildId.toLong()) + .setAudioSender(context.getPlayer(json.getString("guildId"))) } fun stop(session: WebSocketSession, json: JSONObject) { @@ -109,15 +94,7 @@ class WebSocketHandlers(private val contextMap: Map) { } fun destroy(session: WebSocketSession, json: JSONObject) { - val socketContext = contextMap[session.id]!! - val player = socketContext.players.remove(json.getString("guildId")) - player?.stop() - val mem = MagmaMember.builder() - .userId(socketContext.userId) - .guildId(json.getString("guildId")) - .build() - socketContext.magma.removeSendHandler(mem) - socketContext.magma.closeConnection(mem) + contextMap[session.id]!!.destroy(json.getLong("guildId")) } fun configureResuming(session: WebSocketSession, json: JSONObject) { diff --git a/LavalinkServer/src/main/java/lavalink/server/player/Player.java b/LavalinkServer/src/main/java/lavalink/server/player/Player.java index 5ff084ef2..1351845fe 100644 --- a/LavalinkServer/src/main/java/lavalink/server/player/Player.java +++ b/LavalinkServer/src/main/java/lavalink/server/player/Player.java @@ -30,19 +30,20 @@ import com.sedmelluq.discord.lavaplayer.track.AudioTrack; import com.sedmelluq.discord.lavaplayer.track.AudioTrackEndReason; import com.sedmelluq.discord.lavaplayer.track.playback.AudioFrame; +import io.netty.buffer.ByteBuf; import lavalink.server.io.SocketContext; import lavalink.server.io.SocketServer; -import net.dv8tion.jda.api.audio.AudioSendHandler; +import moe.kyokobot.koe.audio.AudioFrameProvider; +import moe.kyokobot.koe.codec.Codec; import org.json.JSONObject; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import javax.annotation.Nullable; -import java.nio.ByteBuffer; import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; -public class Player extends AudioEventAdapter implements AudioSendHandler { +public class Player extends AudioEventAdapter implements AudioFrameProvider { private static final Logger log = LoggerFactory.getLogger(Player.class); @@ -143,7 +144,7 @@ public boolean isPaused() { } @Override - public boolean canProvide() { + public boolean canSendFrame() { lastFrame = player.provide(); if(lastFrame == null) { @@ -156,13 +157,8 @@ public boolean canProvide() { } @Override - public ByteBuffer provide20MsAudio() { - return ByteBuffer.wrap(lastFrame.getData()); - } - - @Override - public boolean isOpus() { - return true; + public void retrieve(Codec codec, ByteBuf buf) { + buf.writeBytes(lastFrame.getData()); } public AudioLossCounter getAudioLossCounter() { diff --git a/build.gradle b/build.gradle index b07980375..3708f1dc0 100644 --- a/build.gradle +++ b/build.gradle @@ -54,7 +54,7 @@ subprojects { //@formatter:off lavaplayerVersion = '1.3.46' lavaplayerIpRotatorVersion = '0.1.7' - koeVersion = '0.2.0' + koeVersion = '7c5fa6e' magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' From e3e2a64a1d9c743f10d59e9c2f3853bdbb463292 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Fri, 10 Apr 2020 14:13:00 +0200 Subject: [PATCH 08/56] Remove Magma and associated dependencies --- LavalinkServer/build.gradle | 5 -- .../config/AudioSendFactoryConfiguration.java | 56 ------------------- .../java/lavalink/server/io/SocketContext.kt | 20 +------ .../java/lavalink/server/io/SocketServer.kt | 8 --- .../lavalink/server/io/WebSocketHandlers.kt | 2 - build.gradle | 3 - 6 files changed, 2 insertions(+), 92 deletions(-) delete mode 100644 LavalinkServer/src/main/java/lavalink/server/config/AudioSendFactoryConfiguration.java diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index a90ee05b0..c9d7b755b 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -36,15 +36,10 @@ bootRun { dependencies { compile group: 'moe.kyokobot', name: 'koe', version: koeVersion compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion - compile group: 'space.npstr.Magma', name: 'magma', version: magmaVersion compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion - compile group: 'com.sedmelluq', name: 'jda-nas', version: jdaNasVersion compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion - compile group: 'com.github.shredder121', name: 'jda-async-packetprovider', version: jappVersion - //required by japp - compile group: 'org.apache.commons', name: 'commons-lang3', version: commonsLangVersion compile group: 'org.springframework', name: 'spring-websocket', version: springWebSocketVersion compile group: 'ch.qos.logback', name: 'logback-classic', version: logbackVersion compile group: 'io.sentry', name: 'sentry-logback', version: sentryLogbackVersion diff --git a/LavalinkServer/src/main/java/lavalink/server/config/AudioSendFactoryConfiguration.java b/LavalinkServer/src/main/java/lavalink/server/config/AudioSendFactoryConfiguration.java deleted file mode 100644 index 31358ec89..000000000 --- a/LavalinkServer/src/main/java/lavalink/server/config/AudioSendFactoryConfiguration.java +++ /dev/null @@ -1,56 +0,0 @@ -package lavalink.server.config; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; -import org.springframework.stereotype.Component; - -/** - * Created by napster on 05.03.18. - */ -@Component -public class AudioSendFactoryConfiguration { - - private static final Logger log = LoggerFactory.getLogger(AudioSendFactoryConfiguration.class); - - private boolean nasSupported = false; - private final int audioSendFactoryCount = Runtime.getRuntime().availableProcessors() * 2; - - public AudioSendFactoryConfiguration(ServerConfig serverConfig) { - String os = System.getProperty("os.name"); - - log.info("OS: " + System.getProperty("os.name") + ", Arch: " + System.getProperty("os.arch")); - - if ((os.contains("Windows") || os.contains("Linux")) - && !System.getProperty("os.arch").equalsIgnoreCase("arm") - && !System.getProperty("os.arch").equalsIgnoreCase("arm-linux") - ) { - nasSupported = true; - log.info("JDA-NAS supported system detected. Enabled native audio sending."); - - Integer customBuffer = serverConfig.getBufferDurationMs(); - if (customBuffer != null) { - log.info("Setting buffer to {}ms", customBuffer); - } else { - log.info("Using default buffer"); - } - - Integer customPlaylistLimit = serverConfig.getYoutubePlaylistLoadLimit(); - if (customPlaylistLimit != null) { - log.info("Setting playlist load limit to {}", customPlaylistLimit); - } else { - log.info("Using default playlist load limit"); - } - } else { - log.warn("This system and architecture appears to not support native audio sending! " - + "GC pauses may cause your bot to stutter during playback."); - } - } - - public boolean isNasSupported() { - return nasSupported; - } - - public int getAudioSendFactoryCount() { - return audioSendFactoryCount; - } -} diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt index 0ddcde51c..1ec19ef26 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt @@ -36,8 +36,6 @@ import org.json.JSONObject import org.slf4j.LoggerFactory import org.springframework.web.socket.WebSocketSession import org.springframework.web.socket.adapter.standard.StandardWebSocketSession -import space.npstr.magma.api.event.MagmaEvent -import space.npstr.magma.api.event.WebSocketClosed import java.util.* import java.util.concurrent.ConcurrentHashMap import java.util.concurrent.ConcurrentLinkedQueue @@ -48,7 +46,7 @@ import java.util.concurrent.TimeUnit class SocketContext internal constructor( val audioPlayerManager: AudioPlayerManager, - var session: WebSocketSession, + private var session: WebSocketSession, private val socketServer: SocketServer, val userId: String, private val koe: KoeClient @@ -99,20 +97,6 @@ class SocketContext internal constructor( return players } - private fun handleMagmaEvent(magmaEvent: MagmaEvent) { - if (magmaEvent is WebSocketClosed) { - val out = JSONObject() - out.put("op", "event") - out.put("type", "WebSocketClosedEvent") - out.put("guildId", magmaEvent.member.guildId) - out.put("reason", magmaEvent.reason) - out.put("code", magmaEvent.closeCode) - out.put("byRemote", magmaEvent.isByRemote) - - send(out) - } - } - /** * Gets or creates a voice connection */ @@ -181,7 +165,7 @@ class SocketContext internal constructor( send(resumeEventQueue.remove()) } - players.values.forEach { it -> SocketServer.sendPlayerUpdate(this, it) } + players.values.forEach { SocketServer.sendPlayerUpdate(this, it) } } internal fun shutdown() { diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt index d8732cff5..e12bf03f5 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt @@ -22,16 +22,11 @@ package lavalink.server.io -import com.github.shredder121.asyncaudio.jda.AsyncPacketProviderFactory -import com.sedmelluq.discord.lavaplayer.jdaudp.NativeAudioSendFactory import com.sedmelluq.discord.lavaplayer.player.AudioPlayerManager -import lavalink.server.config.AudioSendFactoryConfiguration import lavalink.server.config.ServerConfig import lavalink.server.player.Player -import lavalink.server.util.Util import moe.kyokobot.koe.Koe import moe.kyokobot.koe.KoeOptions -import net.dv8tion.jda.api.audio.factory.IAudioSendFactory import org.json.JSONObject import org.slf4j.LoggerFactory import org.springframework.stereotype.Service @@ -39,7 +34,6 @@ import org.springframework.web.socket.CloseStatus import org.springframework.web.socket.TextMessage import org.springframework.web.socket.WebSocketSession import org.springframework.web.socket.handler.TextWebSocketHandler -import space.npstr.magma.api.Member import java.util.* import java.util.concurrent.ConcurrentHashMap @@ -47,14 +41,12 @@ import java.util.concurrent.ConcurrentHashMap class SocketServer( private val serverConfig: ServerConfig, private val audioPlayerManager: AudioPlayerManager, - private val audioSendFactoryConfiguration: AudioSendFactoryConfiguration, koeOptions: KoeOptions ) : TextWebSocketHandler() { // userId <-> shardCount private val shardCounts = ConcurrentHashMap() val contextMap = HashMap() - private val sendFactories = ConcurrentHashMap() @Suppress("LeakingThis") private val handlers = WebSocketHandlers(contextMap) private val resumableSessions = mutableMapOf() diff --git a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt index 95a89f653..18b3ae403 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt @@ -6,8 +6,6 @@ import org.json.JSONObject import org.slf4j.Logger import org.slf4j.LoggerFactory import org.springframework.web.socket.WebSocketSession -import space.npstr.magma.api.MagmaMember -import space.npstr.magma.api.MagmaServerUpdate class WebSocketHandlers(private val contextMap: Map) { diff --git a/build.gradle b/build.gradle index 3708f1dc0..f6fb5d9d7 100644 --- a/build.gradle +++ b/build.gradle @@ -55,9 +55,6 @@ subprojects { lavaplayerVersion = '1.3.46' lavaplayerIpRotatorVersion = '0.1.7' koeVersion = '7c5fa6e' - magmaVersion = '0.12.6' - jdaNasVersion = '1.1.0' - jappVersion = '1.3.2-MINN' springBootVersion = "${springBootVersion}" springWebSocketVersion = '5.1.9.RELEASE' From fba26938b8beed7bc823a0433330c89e7a2f7143 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Fri, 10 Apr 2020 14:55:30 +0200 Subject: [PATCH 09/56] Fix Koe including incompatible SLF4J --- LavalinkServer/build.gradle | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index c9d7b755b..783f666ed 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -34,7 +34,10 @@ bootRun { } dependencies { - compile group: 'moe.kyokobot', name: 'koe', version: koeVersion + compile (group: 'moe.kyokobot.koe', name: 'core', version: koeVersion) { + // This version of SLF4J does not recognise Logback 1.2.3 + exclude group: "org.slf4j", module: "slf4j-api" + } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion From adb43de4dc7c1dc90b7663b62521bd94ef0ecf42 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Fri, 10 Apr 2020 18:33:13 +0200 Subject: [PATCH 10/56] Configure buffer size --- .../main/java/lavalink/server/config/KoeConfiguration.kt | 8 +++++++- build.gradle | 2 +- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt b/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt index d9cf73793..a2563cab3 100644 --- a/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt +++ b/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt @@ -24,7 +24,13 @@ class KoeConfiguration(val serverConfig: ServerConfig) { if (nasSupported) { log.info("Enabling JDA-NAS") - setFramePollerFactory(UdpQueueFramePollerFactory()) + var bufferSize = serverConfig.bufferDurationMs ?: UdpQueueFramePollerFactory.DEFAULT_BUFFER_DURATION + if (bufferSize <= 0) { + log.warn("Buffer size of {}ms is illegal. Defaulting to {}", + bufferSize, UdpQueueFramePollerFactory.DEFAULT_BUFFER_DURATION) + bufferSize = UdpQueueFramePollerFactory.DEFAULT_BUFFER_DURATION + } + setFramePollerFactory(UdpQueueFramePollerFactory(bufferSize)) } else { log.warn("This system and architecture appears to not support native audio sending! " + "GC pauses may cause your bot to stutter during playback.") diff --git a/build.gradle b/build.gradle index f6fb5d9d7..327e0844b 100644 --- a/build.gradle +++ b/build.gradle @@ -54,7 +54,7 @@ subprojects { //@formatter:off lavaplayerVersion = '1.3.46' lavaplayerIpRotatorVersion = '0.1.7' - koeVersion = '7c5fa6e' + koeVersion = '98af1e8' springBootVersion = "${springBootVersion}" springWebSocketVersion = '5.1.9.RELEASE' From fcc1ffe1d05bb3a5b41d5423cc4e441dd0348f15 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sat, 11 Apr 2020 14:21:57 +0200 Subject: [PATCH 11/56] Add Testbot --- .gitignore | 2 + Testbot/build.gradle | 27 +++++ .../main/kotlin/lavalink/testbot/testbot.kt | 100 ++++++++++++++++++ Testbot/src/main/resources/logback.xml | 11 ++ settings.gradle | 1 + 5 files changed, 141 insertions(+) create mode 100644 Testbot/build.gradle create mode 100644 Testbot/src/main/kotlin/lavalink/testbot/testbot.kt create mode 100644 Testbot/src/main/resources/logback.xml diff --git a/.gitignore b/.gitignore index d738fba21..67c6d0d25 100644 --- a/.gitignore +++ b/.gitignore @@ -11,3 +11,5 @@ build/* .project .classpath .settings +/Testbot/build/ +/Testbot/out/ diff --git a/Testbot/build.gradle b/Testbot/build.gradle new file mode 100644 index 000000000..4ad989545 --- /dev/null +++ b/Testbot/build.gradle @@ -0,0 +1,27 @@ +apply plugin: 'kotlin' +apply plugin: 'application' + +group 'lavalink' +version '1.0' + +repositories { + mavenCentral() + jcenter() + maven { url 'https://jitpack.io' } +} + +dependencies { + compile ('net.dv8tion:JDA:4.1.1_135') { + exclude module: 'opus-java' + } + compile "com.github.FredBoat:Lavalink-Client:f84b333518" + compile "ch.qos.logback:logback-classic:$logbackVersion" + compile "org.jetbrains.kotlin:kotlin-stdlib-jdk8" +} + +compileKotlin { + kotlinOptions.jvmTarget = "1.8" +} +compileTestKotlin { + kotlinOptions.jvmTarget = "1.8" +} \ No newline at end of file diff --git a/Testbot/src/main/kotlin/lavalink/testbot/testbot.kt b/Testbot/src/main/kotlin/lavalink/testbot/testbot.kt new file mode 100644 index 000000000..f7e6b598c --- /dev/null +++ b/Testbot/src/main/kotlin/lavalink/testbot/testbot.kt @@ -0,0 +1,100 @@ +package lavalink.testbot + +import com.sedmelluq.discord.lavaplayer.player.AudioLoadResultHandler +import com.sedmelluq.discord.lavaplayer.player.DefaultAudioPlayerManager +import com.sedmelluq.discord.lavaplayer.source.AudioSourceManagers +import com.sedmelluq.discord.lavaplayer.tools.FriendlyException +import com.sedmelluq.discord.lavaplayer.track.AudioPlaylist +import com.sedmelluq.discord.lavaplayer.track.AudioTrack +import lavalink.client.io.jda.JdaLavalink +import net.dv8tion.jda.api.JDA +import net.dv8tion.jda.api.JDABuilder +import net.dv8tion.jda.api.entities.Member +import net.dv8tion.jda.api.entities.TextChannel +import net.dv8tion.jda.api.events.ReadyEvent +import net.dv8tion.jda.api.events.StatusChangeEvent +import net.dv8tion.jda.api.events.message.guild.GuildMessageReceivedEvent +import net.dv8tion.jda.api.hooks.ListenerAdapter +import net.dv8tion.jda.api.requests.GatewayIntent +import net.dv8tion.jda.api.utils.cache.CacheFlag.* +import org.slf4j.Logger +import org.slf4j.LoggerFactory +import java.net.URI + +private val log: Logger = LoggerFactory.getLogger("Testbot") +private lateinit var jda: JDA +private val lavalink = JdaLavalink(1) { _ -> jda } +private val playerManager = DefaultAudioPlayerManager() + +fun main() { + AudioSourceManagers.registerRemoteSources(playerManager) + + jda = JDABuilder.createDefault(System.getenv("TOKEN"), + GatewayIntent.GUILD_MESSAGES, + GatewayIntent.GUILD_VOICE_STATES) + .disableCache(CLIENT_STATUS, ACTIVITY, CLIENT_STATUS, EMOTE, MEMBER_OVERRIDES) + .addEventListeners(Listener, lavalink) + .setVoiceDispatchInterceptor(lavalink.voiceInterceptor) + .build() +} + +object Listener : ListenerAdapter() { + override fun onStatusChange(event: StatusChangeEvent) { + log.info("{} -> {}", event.oldStatus, event.newStatus) + } + + override fun onReady(event: ReadyEvent) { + lavalink.setUserId(jda.selfUser.id) + lavalink.addNode(URI("ws://localhost:2333"), "youshallnotpass") + } + + override fun onGuildMessageReceived(event: GuildMessageReceivedEvent) { + val member = event.member ?: return + if (member.user.isBot) return + + try { + play(event.channel, member, event.message.contentRaw) + } catch (e: Exception) { + event.channel.sendMessage(e.message ?: e.toString()).queue() + } + } +} + +fun play(channel: TextChannel, member: Member, message: String) { + if (message.startsWith(";;play ")) { + val vc = member.voiceState?.channel + if (vc == null) { + channel.sendMessage("You must be in a voice channel").queue() + return + } + + val link = lavalink.getLink(channel.guild) + link.connect(vc) + + val track = message.drop(";;play ".length).trim() + playerManager.loadItem(track, object : AudioLoadResultHandler { + override fun loadFailed(e: FriendlyException) { + channel.sendMessage(e.message ?: e.toString()).queue() + } + + override fun trackLoaded(track: AudioTrack) { + channel.sendMessage("Playing " + track.info.title).queue() + link.player.playTrack(track) + } + + override fun noMatches() { + channel.sendMessage("No matches").queue() + } + + override fun playlistLoaded(playlist: AudioPlaylist) { + val track = playlist.selectedTrack ?: playlist.tracks.firstOrNull() + if (track == null) { + channel.sendMessage("Empty playlist").queue() + return + } + channel.sendMessage("Playing " + track.info.title).queue() + link.player.playTrack(track) + } + }) + } +} \ No newline at end of file diff --git a/Testbot/src/main/resources/logback.xml b/Testbot/src/main/resources/logback.xml new file mode 100644 index 000000000..4bd2e3760 --- /dev/null +++ b/Testbot/src/main/resources/logback.xml @@ -0,0 +1,11 @@ + + + + %d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n + + + + + + + \ No newline at end of file diff --git a/settings.gradle b/settings.gradle index 824cc4b54..a7064b685 100644 --- a/settings.gradle +++ b/settings.gradle @@ -1,4 +1,5 @@ rootProject.name = 'Lavalink-Parent' include ':Lavalink-Server' +include ':Testbot' project(':Lavalink-Server').projectDir = "$rootDir/LavalinkServer" as File From 04ccda1beed1a84e52d5c0ebc31eed51bea545d7 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sat, 11 Apr 2020 14:22:08 +0200 Subject: [PATCH 12/56] Change how audio loss is counted --- LavalinkServer/src/main/java/lavalink/server/player/Player.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/player/Player.java b/LavalinkServer/src/main/java/lavalink/server/player/Player.java index 1351845fe..346b5b6fc 100644 --- a/LavalinkServer/src/main/java/lavalink/server/player/Player.java +++ b/LavalinkServer/src/main/java/lavalink/server/player/Player.java @@ -151,13 +151,13 @@ public boolean canSendFrame() { audioLossCounter.onLoss(); return false; } else { - audioLossCounter.onSuccess(); return true; } } @Override public void retrieve(Codec codec, ByteBuf buf) { + audioLossCounter.onSuccess(); buf.writeBytes(lastFrame.getData()); } From b06e7a526394e2a60d682c92241672888f017459 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sat, 11 Apr 2020 14:55:03 +0200 Subject: [PATCH 13/56] Add testbot main class name --- Testbot/build.gradle | 1 + 1 file changed, 1 insertion(+) diff --git a/Testbot/build.gradle b/Testbot/build.gradle index 4ad989545..82921eb56 100644 --- a/Testbot/build.gradle +++ b/Testbot/build.gradle @@ -3,6 +3,7 @@ apply plugin: 'application' group 'lavalink' version '1.0' +mainClassName = "lavalink.TestbotKt" repositories { mavenCentral() From 37ef3db42527245712170d7d99d46055cada4305 Mon Sep 17 00:00:00 2001 From: Thewsomeguy Date: Sat, 11 Apr 2020 12:27:04 -0500 Subject: [PATCH 14/56] Update build.gradle --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index f70892104..1024f64cf 100644 --- a/build.gradle +++ b/build.gradle @@ -52,7 +52,7 @@ subprojects { ext { //@formatter:off - lavaplayerVersion = '1.3.40' + lavaplayerVersion = '1.3.46' lavaplayerIpRotatorVersion = '0.2.3' magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' From aff38e41548c53ed3fd5a6d46affa7fc8c8e8b86 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Tue, 21 Apr 2020 14:24:39 +0200 Subject: [PATCH 15/56] Update Koe --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index 0747251bd..1c7d94b88 100644 --- a/build.gradle +++ b/build.gradle @@ -57,7 +57,7 @@ subprojects { magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' - koeVersion = '98af1e8' + koeVersion = 'cc528f0' springBootVersion = "${springBootVersion}" springWebSocketVersion = '5.1.9.RELEASE' From 1d8e3e3285e8a084b25b2d0353a0a0254f24ea9c Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Tue, 21 Apr 2020 14:48:23 +0200 Subject: [PATCH 16/56] Fix bad Koe version name --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index 1c7d94b88..7e989fec9 100644 --- a/build.gradle +++ b/build.gradle @@ -57,7 +57,7 @@ subprojects { magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' - koeVersion = 'cc528f0' + koeVersion = 'cc528f03' springBootVersion = "${springBootVersion}" springWebSocketVersion = '5.1.9.RELEASE' From c66910e6bd63b985ba25baa162b79db374917f79 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sat, 9 May 2020 23:11:07 +0200 Subject: [PATCH 17/56] Fix Koe breaking changes --- .../server/config/KoeConfiguration.kt | 2 +- .../lavalink/server/io/WebSocketHandlers.kt | 4 +- .../java/lavalink/server/player/Player.java | 60 +++++++++++-------- 3 files changed, 38 insertions(+), 28 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt b/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt index a2563cab3..ef9b08d44 100644 --- a/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt +++ b/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt @@ -30,7 +30,7 @@ class KoeConfiguration(val serverConfig: ServerConfig) { bufferSize, UdpQueueFramePollerFactory.DEFAULT_BUFFER_DURATION) bufferSize = UdpQueueFramePollerFactory.DEFAULT_BUFFER_DURATION } - setFramePollerFactory(UdpQueueFramePollerFactory(bufferSize)) + setFramePollerFactory(UdpQueueFramePollerFactory()) } else { log.warn("This system and architecture appears to not support native audio sending! " + "GC pauses may cause your bot to stutter during playback.") diff --git a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt index 18b3ae403..5e0830f71 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt @@ -53,8 +53,8 @@ class WebSocketHandlers(private val contextMap: Map) { val context = contextMap[session.id]!! - context.getVoiceConnection(player.guildId.toLong()) - .setAudioSender(context.getPlayer(json.getString("guildId"))) + val conn = context.getVoiceConnection(player.guildId.toLong()) + context.getPlayer(json.getString("guildId")).provideTo(conn) } fun stop(session: WebSocketSession, json: JSONObject) { diff --git a/LavalinkServer/src/main/java/lavalink/server/player/Player.java b/LavalinkServer/src/main/java/lavalink/server/player/Player.java index 346b5b6fc..f91d23987 100644 --- a/LavalinkServer/src/main/java/lavalink/server/player/Player.java +++ b/LavalinkServer/src/main/java/lavalink/server/player/Player.java @@ -33,8 +33,8 @@ import io.netty.buffer.ByteBuf; import lavalink.server.io.SocketContext; import lavalink.server.io.SocketServer; -import moe.kyokobot.koe.audio.AudioFrameProvider; -import moe.kyokobot.koe.codec.Codec; +import moe.kyokobot.koe.VoiceConnection; +import moe.kyokobot.koe.media.OpusAudioFrameProvider; import org.json.JSONObject; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -43,17 +43,17 @@ import java.util.concurrent.ScheduledFuture; import java.util.concurrent.TimeUnit; -public class Player extends AudioEventAdapter implements AudioFrameProvider { +public class Player extends AudioEventAdapter { private static final Logger log = LoggerFactory.getLogger(Player.class); - private SocketContext socketContext; + private final SocketContext socketContext; private final String guildId; private final AudioPlayer player; - private AudioLossCounter audioLossCounter = new AudioLossCounter(); + private final AudioLossCounter audioLossCounter = new AudioLossCounter(); private AudioFrame lastFrame = null; - private ScheduledFuture myFuture = null; - private EqualizerFactory equalizerFactory = new EqualizerFactory(); + private ScheduledFuture myFuture = null; + private final EqualizerFactory equalizerFactory = new EqualizerFactory(); private boolean isEqualizerApplied = false; public Player(SocketContext socketContext, String guildId, AudioPlayerManager audioPlayerManager) { @@ -143,24 +143,6 @@ public boolean isPaused() { return player.isPaused(); } - @Override - public boolean canSendFrame() { - lastFrame = player.provide(); - - if(lastFrame == null) { - audioLossCounter.onLoss(); - return false; - } else { - return true; - } - } - - @Override - public void retrieve(Codec codec, ByteBuf buf) { - audioLossCounter.onSuccess(); - buf.writeBytes(lastFrame.getData()); - } - public AudioLossCounter getAudioLossCounter() { return audioLossCounter; } @@ -185,4 +167,32 @@ public void onTrackStart(AudioPlayer player, AudioTrack track) { } } + public void provideTo(VoiceConnection connection) { + new Provider(connection); + } + + private class Provider extends OpusAudioFrameProvider { + public Provider(VoiceConnection connection) { + super(connection); + } + + @Override + public boolean canProvide() { + lastFrame = player.provide(); + + if(lastFrame == null) { + audioLossCounter.onLoss(); + return false; + } else { + return true; + } + } + + @Override + public void retrieveOpusFrame(ByteBuf buf) { + audioLossCounter.onSuccess(); + buf.writeBytes(lastFrame.getData()); + } + } + } From adc42cd8c16721639e7b0c7fcf4f39401a302519 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sat, 9 May 2020 23:17:26 +0200 Subject: [PATCH 18/56] Refactor WS handling --- .../java/lavalink/server/io/SocketServer.kt | 21 +++++---- .../lavalink/server/io/WebSocketHandlers.kt | 43 ++++++++----------- 2 files changed, 30 insertions(+), 34 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt index e12bf03f5..f93309258 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt @@ -142,17 +142,20 @@ class SocketServer( return } + val context = contextMap[session.id] + ?: throw IllegalStateException("No context for session ID ${session.id}. Broken websocket?") + when (json.getString("op")) { // @formatter:off - "voiceUpdate" -> handlers.voiceUpdate(session, json) - "play" -> handlers.play(session, json) - "stop" -> handlers.stop(session, json) - "pause" -> handlers.pause(session, json) - "seek" -> handlers.seek(session, json) - "volume" -> handlers.volume(session, json) - "destroy" -> handlers.destroy(session, json) - "configureResuming" -> handlers.configureResuming(session, json) - "equalizer" -> handlers.equalizer(session, json) + "voiceUpdate" -> handlers.voiceUpdate(context, json) + "play" -> handlers.play(context, json) + "stop" -> handlers.stop(context, json) + "pause" -> handlers.pause(context, json) + "seek" -> handlers.seek(context, json) + "volume" -> handlers.volume(context, json) + "destroy" -> handlers.destroy(context, json) + "configureResuming" -> handlers.configureResuming(context, json) + "equalizer" -> handlers.equalizer(context, json) else -> log.warn("Unexpected operation: " + json.getString("op")) // @formatter:on } diff --git a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt index 5e0830f71..8b94398e5 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt @@ -13,7 +13,7 @@ class WebSocketHandlers(private val contextMap: Map) { private val log: Logger = LoggerFactory.getLogger(WebSocketHandlers::class.java) } - fun voiceUpdate(session: WebSocketSession, json: JSONObject) { + fun voiceUpdate(context: SocketContext, json: JSONObject) { val sessionId = json.getString("sessionId") val guildId = json.getLong("guildId") @@ -24,13 +24,11 @@ class WebSocketHandlers(private val contextMap: Map) { //discord sometimes send a partial server update missing the endpoint, which can be ignored. endpoint ?: return - val sktContext = contextMap[session.id]!! - sktContext.getVoiceConnection(guildId).connect(VoiceServerInfo(sessionId, endpoint, token)); + context.getVoiceConnection(guildId).connect(VoiceServerInfo(sessionId, endpoint, token)); } - fun play(session: WebSocketSession, json: JSONObject) { - val ctx = contextMap[session.id]!! - val player = ctx.getPlayer(json.getString("guildId")) + fun play(context: SocketContext, json: JSONObject) { + val player = context.getPlayer(json.getString("guildId")) val noReplace = json.optBoolean("noReplace", false) if (noReplace && player.playingTrack != null) { @@ -38,7 +36,7 @@ class WebSocketHandlers(private val contextMap: Map) { return } - val track = Util.toAudioTrack(ctx.audioPlayerManager, json.getString("track")) + val track = Util.toAudioTrack(context.audioPlayerManager, json.getString("track")) if (json.has("startTime")) { track.position = json.getLong("startTime") @@ -51,38 +49,34 @@ class WebSocketHandlers(private val contextMap: Map) { player.play(track) - val context = contextMap[session.id]!! - val conn = context.getVoiceConnection(player.guildId.toLong()) context.getPlayer(json.getString("guildId")).provideTo(conn) } - fun stop(session: WebSocketSession, json: JSONObject) { - val player = contextMap[session.id]!!.getPlayer(json.getString("guildId")) + fun stop(context: SocketContext, json: JSONObject) { + val player = context.getPlayer(json.getString("guildId")) player.stop() } - fun pause(session: WebSocketSession, json: JSONObject) { - val context = contextMap[session.id]!! + fun pause(context: SocketContext, json: JSONObject) { val player = context.getPlayer(json.getString("guildId")) player.setPause(json.getBoolean("pause")) SocketServer.sendPlayerUpdate(context, player) } - fun seek(session: WebSocketSession, json: JSONObject) { - val context = contextMap[session.id]!! + fun seek(context: SocketContext, json: JSONObject) { val player = context.getPlayer(json.getString("guildId")) player.seekTo(json.getLong("position")) SocketServer.sendPlayerUpdate(context, player) } - fun volume(session: WebSocketSession, json: JSONObject) { - val player = contextMap[session.id]!!.getPlayer(json.getString("guildId")) + fun volume(context: SocketContext, json: JSONObject) { + val player = context.getPlayer(json.getString("guildId")) player.setVolume(json.getInt("volume")) } - fun equalizer(session: WebSocketSession, json: JSONObject) { - val player = contextMap[session.id]!!.getPlayer(json.getString("guildId")) + fun equalizer(context: SocketContext, json: JSONObject) { + val player = context.getPlayer(json.getString("guildId")) val bands = json.getJSONArray("bands") for (i in 0 until bands.length()) { @@ -91,13 +85,12 @@ class WebSocketHandlers(private val contextMap: Map) { } } - fun destroy(session: WebSocketSession, json: JSONObject) { - contextMap[session.id]!!.destroy(json.getLong("guildId")) + fun destroy(context: SocketContext, json: JSONObject) { + context.destroy(json.getLong("guildId")) } - fun configureResuming(session: WebSocketSession, json: JSONObject) { - val socketContext = contextMap[session.id]!! - socketContext.resumeKey = json.optString("key", null) - if (json.has("timeout")) socketContext.resumeTimeout = json.getLong("timeout") + fun configureResuming(context: SocketContext, json: JSONObject) { + context.resumeKey = json.optString("key", null) + if (json.has("timeout")) context.resumeTimeout = json.getLong("timeout") } } \ No newline at end of file From d458f0da2a1a69d8e69e0f1888f361ac030a9fd6 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sun, 10 May 2020 12:05:35 +0200 Subject: [PATCH 19/56] Fix not setting audio sender Co-authored-by: Antoine L. --- LavalinkServer/src/main/java/lavalink/server/player/Player.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/player/Player.java b/LavalinkServer/src/main/java/lavalink/server/player/Player.java index f91d23987..a63f0fa5f 100644 --- a/LavalinkServer/src/main/java/lavalink/server/player/Player.java +++ b/LavalinkServer/src/main/java/lavalink/server/player/Player.java @@ -168,7 +168,7 @@ public void onTrackStart(AudioPlayer player, AudioTrack track) { } public void provideTo(VoiceConnection connection) { - new Provider(connection); + connection.setAudioSender(new Provider(connection)); } private class Provider extends OpusAudioFrameProvider { From 9a8c3f74dc4c4f7158b608b4f49254ca204e3a0d Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sat, 9 May 2020 23:19:25 +0200 Subject: [PATCH 20/56] Configure koe bufferSize --- .../src/main/java/lavalink/server/config/KoeConfiguration.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt b/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt index ef9b08d44..d787a3ed7 100644 --- a/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt +++ b/LavalinkServer/src/main/java/lavalink/server/config/KoeConfiguration.kt @@ -30,7 +30,7 @@ class KoeConfiguration(val serverConfig: ServerConfig) { bufferSize, UdpQueueFramePollerFactory.DEFAULT_BUFFER_DURATION) bufferSize = UdpQueueFramePollerFactory.DEFAULT_BUFFER_DURATION } - setFramePollerFactory(UdpQueueFramePollerFactory()) + setFramePollerFactory(UdpQueueFramePollerFactory(bufferSize, Runtime.getRuntime().availableProcessors())) } else { log.warn("This system and architecture appears to not support native audio sending! " + "GC pauses may cause your bot to stutter during playback.") From 539781e2ba7197b00c953a67f61776cc69297cfb Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Mon, 11 May 2020 15:15:07 +0200 Subject: [PATCH 21/56] Fix not immediately sending player update on play --- LavalinkServer/src/main/java/lavalink/server/player/Player.java | 1 + 1 file changed, 1 insertion(+) diff --git a/LavalinkServer/src/main/java/lavalink/server/player/Player.java b/LavalinkServer/src/main/java/lavalink/server/player/Player.java index a63f0fa5f..8caf16641 100644 --- a/LavalinkServer/src/main/java/lavalink/server/player/Player.java +++ b/LavalinkServer/src/main/java/lavalink/server/player/Player.java @@ -67,6 +67,7 @@ public Player(SocketContext socketContext, String guildId, AudioPlayerManager au public void play(AudioTrack track) { player.playTrack(track); + SocketServer.Companion.sendPlayerUpdate(socketContext, this); } public void stop() { From 9085045dcdc00078036fede6cf806e8b45cfa405 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Mon, 11 May 2020 22:37:05 +0200 Subject: [PATCH 22/56] Handle koe null WS close reason --- .../src/main/java/lavalink/server/io/SocketContext.kt | 2 +- build.gradle | 1 - 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt index 1ec19ef26..ebe2ae291 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt @@ -182,7 +182,7 @@ class SocketContext internal constructor( out.put("op", "event") out.put("type", "WebSocketClosedEvent") out.put("guildId", guildId) - out.put("reason", reason) + out.put("reason", reason ?: "") out.put("code", code) out.put("byRemote", byRemote) diff --git a/build.gradle b/build.gradle index 7e989fec9..f3afe3b49 100644 --- a/build.gradle +++ b/build.gradle @@ -54,7 +54,6 @@ subprojects { //@formatter:off lavaplayerVersion = '1.3.46' lavaplayerIpRotatorVersion = '0.2.3' - magmaVersion = '0.12.6' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' koeVersion = 'cc528f03' From 89ee04761f0ebdf9925b568addbc9e9d84ba16f8 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Tue, 12 May 2020 19:05:31 +0200 Subject: [PATCH 23/56] Update lavaplayer to 1.3.48 --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index f3afe3b49..027bcf415 100644 --- a/build.gradle +++ b/build.gradle @@ -52,7 +52,7 @@ subprojects { ext { //@formatter:off - lavaplayerVersion = '1.3.46' + lavaplayerVersion = '1.3.48' lavaplayerIpRotatorVersion = '0.2.3' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' From f26a6b28c4e69ddefa16b91cbbe8375c80e16dcf Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Tue, 12 May 2020 19:29:18 +0200 Subject: [PATCH 24/56] Remove NAS attribution, remove V2 clients --- README.md | 9 +-------- 1 file changed, 1 insertion(+), 8 deletions(-) diff --git a/README.md b/README.md index 9399f009e..026c1b6d6 100644 --- a/README.md +++ b/README.md @@ -68,7 +68,6 @@ users about the compatibility of their clients to the Lavalink server. ## Client libraries: -### Supports 3.x and older: * [Lavalink-Client](https://github.com/FredBoat/Lavalink-Client) (JDA or generic, Java) * [LavaClient](https://github.com/SamOphis/LavaClient) (Java) * [Lavalink.py](https://github.com/Devoxin/Lavalink.py) (discord.py, Python) @@ -88,10 +87,6 @@ users about the compatibility of their clients to the Lavalink server. * [WaveLink](https://github.com/EvieePy/Wavelink)(discord.py, Python) * Or [create your own](https://github.com/Frederikam/Lavalink/blob/master/IMPLEMENTATION.md) -### Supports 2.x: -* [eris-lavalink](https://github.com/briantanner/eris-lavalink) (Eris, JavaScript) -* Or [create your own](https://github.com/Frederikam/Lavalink/blob/master/IMPLEMENTATION.md) - ## Server configuration Download binaries from [the CI server](https://ci.fredboat.com/viewLog.html?buildId=lastSuccessful&buildTypeId=Lavalink_Build&tab=artifacts&guest=1) or [the GitHub releases](https://github.com/Frederikam/Lavalink/releases). @@ -102,6 +97,4 @@ Run with `java -jar Lavalink.jar` Docker images are available on the [Docker hub](https://hub.docker.com/r/fredboat/lavalink/). [![Docker Pulls](https://img.shields.io/docker/pulls/fredboat/lavalink.svg)](https://hub.docker.com/r/fredboat/lavalink/) [![Docker layers](https://images.microbadger.com/badges/image/fredboat/lavalink:dev.svg)](https://microbadger.com/images/fredboat/lavalink:dev "Get your own image badge on microbadger.com") - -# Acknowledgements -This project contains modified code from https://github.com/sedmelluq/jda-nas v1.0.5 +notice From 863928f4e6c50735dfc568d55f550d8d274f49db Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Tue, 12 May 2020 19:30:09 +0200 Subject: [PATCH 25/56] Fix dangling word --- README.md | 1 - 1 file changed, 1 deletion(-) diff --git a/README.md b/README.md index 026c1b6d6..7d37cc67a 100644 --- a/README.md +++ b/README.md @@ -97,4 +97,3 @@ Run with `java -jar Lavalink.jar` Docker images are available on the [Docker hub](https://hub.docker.com/r/fredboat/lavalink/). [![Docker Pulls](https://img.shields.io/docker/pulls/fredboat/lavalink.svg)](https://hub.docker.com/r/fredboat/lavalink/) [![Docker layers](https://images.microbadger.com/badges/image/fredboat/lavalink:dev.svg)](https://microbadger.com/images/fredboat/lavalink:dev "Get your own image badge on microbadger.com") -notice From 420247ec35edd9fc2fd75b340a5425e2b6411b63 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Thu, 14 May 2020 11:37:37 +0200 Subject: [PATCH 26/56] Update Koe Might fix green circle bug https://github.com/KyokoBot/koe/issues/10 --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index 027bcf415..d7f3a09b7 100644 --- a/build.gradle +++ b/build.gradle @@ -56,7 +56,7 @@ subprojects { lavaplayerIpRotatorVersion = '0.2.3' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' - koeVersion = 'cc528f03' + koeVersion = '1.0.0-beta3' springBootVersion = "${springBootVersion}" springWebSocketVersion = '5.1.9.RELEASE' From 2328102657c0b17a036fca4544feacf05d239055 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Mon, 18 May 2020 14:35:32 +0200 Subject: [PATCH 27/56] Made some QoL changes to Testbot --- Testbot/build.gradle | 2 +- .../main/kotlin/lavalink/testbot/testbot.kt | 26 +++++++++++++------ 2 files changed, 19 insertions(+), 9 deletions(-) diff --git a/Testbot/build.gradle b/Testbot/build.gradle index 82921eb56..8c9c46d4b 100644 --- a/Testbot/build.gradle +++ b/Testbot/build.gradle @@ -3,7 +3,7 @@ apply plugin: 'application' group 'lavalink' version '1.0' -mainClassName = "lavalink.TestbotKt" +mainClassName = "lavalink.testbot.TestbotKt" repositories { mavenCentral() diff --git a/Testbot/src/main/kotlin/lavalink/testbot/testbot.kt b/Testbot/src/main/kotlin/lavalink/testbot/testbot.kt index f7e6b598c..0826b261d 100644 --- a/Testbot/src/main/kotlin/lavalink/testbot/testbot.kt +++ b/Testbot/src/main/kotlin/lavalink/testbot/testbot.kt @@ -19,17 +19,27 @@ import net.dv8tion.jda.api.requests.GatewayIntent import net.dv8tion.jda.api.utils.cache.CacheFlag.* import org.slf4j.Logger import org.slf4j.LoggerFactory +import java.lang.IllegalArgumentException import java.net.URI private val log: Logger = LoggerFactory.getLogger("Testbot") private lateinit var jda: JDA private val lavalink = JdaLavalink(1) { _ -> jda } private val playerManager = DefaultAudioPlayerManager() +lateinit var host: String +lateinit var password: String -fun main() { +fun main(args: Array) { + if (args.size < 3) { + throw IllegalArgumentException("Expected 3 arguments. Please refer to the readme.") + } + + val token = args[0] + host = args[1] + password = args[2] AudioSourceManagers.registerRemoteSources(playerManager) - jda = JDABuilder.createDefault(System.getenv("TOKEN"), + jda = JDABuilder.createDefault(token, GatewayIntent.GUILD_MESSAGES, GatewayIntent.GUILD_VOICE_STATES) .disableCache(CLIENT_STATUS, ACTIVITY, CLIENT_STATUS, EMOTE, MEMBER_OVERRIDES) @@ -45,7 +55,7 @@ object Listener : ListenerAdapter() { override fun onReady(event: ReadyEvent) { lavalink.setUserId(jda.selfUser.id) - lavalink.addNode(URI("ws://localhost:2333"), "youshallnotpass") + lavalink.addNode(URI(host), password) } override fun onGuildMessageReceived(event: GuildMessageReceivedEvent) { @@ -78,7 +88,7 @@ fun play(channel: TextChannel, member: Member, message: String) { } override fun trackLoaded(track: AudioTrack) { - channel.sendMessage("Playing " + track.info.title).queue() + channel.sendMessage("Playing `${track.info.title}`").queue() link.player.playTrack(track) } @@ -87,13 +97,13 @@ fun play(channel: TextChannel, member: Member, message: String) { } override fun playlistLoaded(playlist: AudioPlaylist) { - val track = playlist.selectedTrack ?: playlist.tracks.firstOrNull() - if (track == null) { + val loaded = playlist.selectedTrack ?: playlist.tracks.firstOrNull() + if (loaded == null) { channel.sendMessage("Empty playlist").queue() return } - channel.sendMessage("Playing " + track.info.title).queue() - link.player.playTrack(track) + channel.sendMessage("Playing `${loaded.info.title}` from list `${playlist.name}`").queue() + link.player.playTrack(loaded) } }) } From b4d61d3078fa33ee36344435c167e8258b00ca1a Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Mon, 18 May 2020 14:35:47 +0200 Subject: [PATCH 28/56] Document Testbot --- README.md | 2 ++ Testbot/readme.md | 35 +++++++++++++++++++++++++++++++++++ 2 files changed, 37 insertions(+) create mode 100644 Testbot/readme.md diff --git a/README.md b/README.md index 8364c9517..1c78a26cd 100644 --- a/README.md +++ b/README.md @@ -4,6 +4,8 @@ Allows for sending audio without it ever reaching any of your shards. Being used in production by FredBoat, Dyno, Rythm, LewdBot, and more. +A [very simple example bot](Testbot) is available. + [![JDA guild](https://discordapp.com/api/guilds/125227483518861312/embed.png?style=banner2)](https://discord.gg/jtAWrzU) ## Features diff --git a/Testbot/readme.md b/Testbot/readme.md new file mode 100644 index 000000000..96212c143 --- /dev/null +++ b/Testbot/readme.md @@ -0,0 +1,35 @@ +# Lavalink Testbot +This is a minimalistic example of a bot using Lavalink. This example is based upon Lavalink-Client for Java. + +https://github.com/FredBoat/Lavalink-Client + +This uses Lavaplayer to load tracks rather than initially loading tracks via Lavalink. Non-JVM (Java) bots will need +to query the `/loadtracks` endpoint. + + +## Running the test bot +This guide assumes the following: +* You have a Lavalink node already running +* You have a Discord bot and its token +* You have a shell in the same directory as this readme +* You have Java 11 or newer. Check with `java -version` + +Run this command. Gradle will download as needed, and will build and run the bot: +```bash +../gradlew run --args "YOUR_BOT_TOKEN ws://localhost:2333 youshallnotpass" +``` + +Replace token, host, and password as needed. + +Replace `../gradlew` with `../gradlew.bat` if on Windows. + + +## Using the test bot +Only one command is currently supported, and only the first or selected track of a playlist or search will be played. Example usages: +``` +;;play https://www.youtube.com/watch?v=cRh1-_pRDzo +;;play https://soundcloud.com/home-2001/resonance +;;play https://www.youtube.com/watch?v=t2D5HlKLh34&list=PLKUyqLlH6brkzzJgD6Gdriga4mdtCAMBJ +;;play ytsearch: John Coltrane Giant Steps +;;play https://www.youtube.com/watch?v=dQw4w9WgXcQ +``` \ No newline at end of file From efcc4c423e2f7d279f67b94e0ac2e19266f2d020 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Mon, 18 May 2020 14:57:38 +0200 Subject: [PATCH 29/56] Add info statement about https://github.com/Frederikam/Lavalink/issues/295 --- LavalinkServer/src/main/java/lavalink/server/Launcher.kt | 1 + 1 file changed, 1 insertion(+) diff --git a/LavalinkServer/src/main/java/lavalink/server/Launcher.kt b/LavalinkServer/src/main/java/lavalink/server/Launcher.kt index 648b281fe..b1a83d26e 100644 --- a/LavalinkServer/src/main/java/lavalink/server/Launcher.kt +++ b/LavalinkServer/src/main/java/lavalink/server/Launcher.kt @@ -125,5 +125,6 @@ object Launcher { } ) sa.run(*args) + log.info("You can safely ignore the big red warning about illegal reflection. See https://github.com/Frederikam/Lavalink/issues/295") } } From ddb5f73b218ca0fc5f494788b282856e3728c6cd Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Fri, 22 May 2020 17:09:21 +0200 Subject: [PATCH 30/56] Lavaplayer -> 1.3.49 --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index d7f3a09b7..992e1c9b4 100644 --- a/build.gradle +++ b/build.gradle @@ -52,7 +52,7 @@ subprojects { ext { //@formatter:off - lavaplayerVersion = '1.3.48' + lavaplayerVersion = '1.3.49' lavaplayerIpRotatorVersion = '0.2.3' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' From 37e84f6752b5600f57f2b40a014db7eb21f9de40 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Mon, 15 Jun 2020 16:17:33 +0200 Subject: [PATCH 31/56] Add improved rest track loader logging --- .../lavalink/server/player/AudioLoaderRestHandler.java | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/player/AudioLoaderRestHandler.java b/LavalinkServer/src/main/java/lavalink/server/player/AudioLoaderRestHandler.java index ee24cc7c6..de120f96c 100644 --- a/LavalinkServer/src/main/java/lavalink/server/player/AudioLoaderRestHandler.java +++ b/LavalinkServer/src/main/java/lavalink/server/player/AudioLoaderRestHandler.java @@ -106,6 +106,7 @@ private JSONObject encodeLoadResult(LoadResult result) { exception.put("severity", result.exception.severity.toString()); json.put("exception", exception); + log.error("Track loading failed", result.exception); } return json; @@ -113,10 +114,10 @@ private JSONObject encodeLoadResult(LoadResult result) { @GetMapping(value = "/loadtracks", produces = "application/json") @ResponseBody - public CompletionStage> getLoadTracks(HttpServletRequest request, - @RequestParam String identifier) { - - log(request); + public CompletionStage> getLoadTracks( + HttpServletRequest request, + @RequestParam String identifier) { + log.info("Got request to load for identifier \"{}\"", identifier); return new AudioLoader(audioPlayerManager).load(identifier) .thenApply(this::encodeLoadResult) From c6c74f6d3be2085f5d699412633ea028cffc19b8 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Wed, 17 Jun 2020 00:37:23 +0200 Subject: [PATCH 32/56] Use patched lavaplayer Includes my patch which should fix search results only working approximately half the time. --- LavalinkServer/build.gradle | 6 ++++-- build.gradle | 3 ++- 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index 783f666ed..ab1f9827b 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -39,8 +39,10 @@ dependencies { exclude group: "org.slf4j", module: "slf4j-api" } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion - compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion - compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion + compile group: 'com.github.FredBoat', name: 'lavaplayer', version: "0d91444efb215dc98e99199de8eb4389c74c1f58" + compile (group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion) { + exclude group: "com.sedmelluq", module: "lavaplayer" + } compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion compile group: 'org.springframework', name: 'spring-websocket', version: springWebSocketVersion diff --git a/build.gradle b/build.gradle index 992e1c9b4..d191b9e9d 100644 --- a/build.gradle +++ b/build.gradle @@ -7,6 +7,7 @@ buildscript { testLoggerVersion = '1.6.0' } repositories { + mavenLocal() maven { url "https://plugins.gradle.org/m2/" } maven { url 'http://repo.spring.io/plugins-release' } maven { url 'https://jitpack.io' } @@ -52,7 +53,7 @@ subprojects { ext { //@formatter:off - lavaplayerVersion = '1.3.49' + lavaplayerVersion = '-SNAPSHOT' lavaplayerIpRotatorVersion = '0.2.3' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' From 123eb9405903b39630b707f495492ed98b8ea63e Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Wed, 17 Jun 2020 21:30:44 +0200 Subject: [PATCH 33/56] Update lavaplayer to 1.3.50 --- LavalinkServer/build.gradle | 6 ++---- build.gradle | 2 +- 2 files changed, 3 insertions(+), 5 deletions(-) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index ab1f9827b..783f666ed 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -39,10 +39,8 @@ dependencies { exclude group: "org.slf4j", module: "slf4j-api" } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion - compile group: 'com.github.FredBoat', name: 'lavaplayer', version: "0d91444efb215dc98e99199de8eb4389c74c1f58" - compile (group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion) { - exclude group: "com.sedmelluq", module: "lavaplayer" - } + compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion + compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion compile group: 'org.springframework', name: 'spring-websocket', version: springWebSocketVersion diff --git a/build.gradle b/build.gradle index d191b9e9d..3674fcf80 100644 --- a/build.gradle +++ b/build.gradle @@ -53,7 +53,7 @@ subprojects { ext { //@formatter:off - lavaplayerVersion = '-SNAPSHOT' + lavaplayerVersion = '1.3.50' lavaplayerIpRotatorVersion = '0.2.3' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' From 7a29ad087983dbab379af0129f5ac0358e2d2899 Mon Sep 17 00:00:00 2001 From: Neuheit <38368299+Neuheit@users.noreply.github.com> Date: Thu, 18 Jun 2020 16:48:00 -0400 Subject: [PATCH 34/56] Added support for stopTime. --- .../java/lavalink/server/io/WebSocketHandlers.kt | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt index 8b94398e5..b53576597 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt @@ -1,11 +1,12 @@ package lavalink.server.io +import com.sedmelluq.discord.lavaplayer.track.TrackMarker +import lavalink.server.player.TrackEndMarkerHandler import lavalink.server.util.Util import moe.kyokobot.koe.VoiceServerInfo import org.json.JSONObject import org.slf4j.Logger import org.slf4j.LoggerFactory -import org.springframework.web.socket.WebSocketSession class WebSocketHandlers(private val contextMap: Map) { @@ -24,7 +25,7 @@ class WebSocketHandlers(private val contextMap: Map) { //discord sometimes send a partial server update missing the endpoint, which can be ignored. endpoint ?: return - context.getVoiceConnection(guildId).connect(VoiceServerInfo(sessionId, endpoint, token)); + context.getVoiceConnection(guildId).connect(VoiceServerInfo(sessionId, endpoint, token)) } fun play(context: SocketContext, json: JSONObject) { @@ -47,6 +48,16 @@ class WebSocketHandlers(private val contextMap: Map) { player.setVolume(json.getInt("volume")) } + if (json.has("stopTime")) { + + val stopTime = json.getLong("stopTime") + if (stopTime > 0) { + val handler = TrackEndMarkerHandler(player) + val marker = TrackMarker(stopTime, handler) + track.setMarker(marker) + } + } + player.play(track) val conn = context.getVoiceConnection(player.guildId.toLong()) From cc67386f5eb38d7a150d668b8a2e7b9c9af5458c Mon Sep 17 00:00:00 2001 From: Neuheit <38368299+Neuheit@users.noreply.github.com> Date: Thu, 18 Jun 2020 16:50:27 -0400 Subject: [PATCH 35/56] Adjust spacing. --- .../src/main/java/lavalink/server/io/WebSocketHandlers.kt | 1 - 1 file changed, 1 deletion(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt index b53576597..8f59cd591 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt @@ -49,7 +49,6 @@ class WebSocketHandlers(private val contextMap: Map) { } if (json.has("stopTime")) { - val stopTime = json.getLong("stopTime") if (stopTime > 0) { val handler = TrackEndMarkerHandler(player) From 43b79fa4a9ab722862e18ef5ab008ca757844703 Mon Sep 17 00:00:00 2001 From: Neuheit <38368299+Neuheit@users.noreply.github.com> Date: Thu, 18 Jun 2020 17:33:59 -0400 Subject: [PATCH 36/56] Feedback. --- .../src/main/java/lavalink/server/io/WebSocketHandlers.kt | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt index 8f59cd591..9080c566c 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/WebSocketHandlers.kt @@ -48,8 +48,8 @@ class WebSocketHandlers(private val contextMap: Map) { player.setVolume(json.getInt("volume")) } - if (json.has("stopTime")) { - val stopTime = json.getLong("stopTime") + if (json.has("endTime")) { + val stopTime = json.getLong("endTime") if (stopTime > 0) { val handler = TrackEndMarkerHandler(player) val marker = TrackMarker(stopTime, handler) @@ -103,4 +103,4 @@ class WebSocketHandlers(private val contextMap: Map) { context.resumeKey = json.optString("key", null) if (json.has("timeout")) context.resumeTimeout = json.getLong("timeout") } -} \ No newline at end of file +} From 069052f315e158dbb912e8fc64367deb69b20d96 Mon Sep 17 00:00:00 2001 From: TheEssem Date: Mon, 6 Jul 2020 10:51:41 -0500 Subject: [PATCH 37/56] Very bad code to add a playerUpdate interval option in config --- LavalinkServer/application.yml.example | 1 + .../main/java/lavalink/server/config/ServerConfig.kt | 1 + .../src/main/java/lavalink/server/io/SocketContext.kt | 4 +++- .../src/main/java/lavalink/server/io/SocketServer.kt | 1 + .../src/main/java/lavalink/server/player/Player.java | 11 +++++++++-- 5 files changed, 15 insertions(+), 3 deletions(-) diff --git a/LavalinkServer/application.yml.example b/LavalinkServer/application.yml.example index 27247bed5..2838b310e 100644 --- a/LavalinkServer/application.yml.example +++ b/LavalinkServer/application.yml.example @@ -15,6 +15,7 @@ lavalink: local: false bufferDurationMs: 400 youtubePlaylistLoadLimit: 6 # Number of pages at 100 each + playerUpdateInterval: 5 youtubeSearchEnabled: true soundcloudSearchEnabled: true gc-warnings: true diff --git a/LavalinkServer/src/main/java/lavalink/server/config/ServerConfig.kt b/LavalinkServer/src/main/java/lavalink/server/config/ServerConfig.kt index 7df6e1926..84c043d19 100644 --- a/LavalinkServer/src/main/java/lavalink/server/config/ServerConfig.kt +++ b/LavalinkServer/src/main/java/lavalink/server/config/ServerConfig.kt @@ -33,6 +33,7 @@ class ServerConfig { var sentryDsn = "" var bufferDurationMs: Int? = null var youtubePlaylistLoadLimit: Int? = null + var playerUpdateInterval: Int? = null var isGcWarnings = true var isYoutubeSearchEnabled = true var isSoundcloudSearchEnabled = true diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt index ebe2ae291..749ac526d 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketContext.kt @@ -28,6 +28,7 @@ import io.undertow.websockets.core.WebSocketChannel import io.undertow.websockets.core.WebSockets import io.undertow.websockets.jsr.UndertowSession import lavalink.server.player.Player +import lavalink.server.config.ServerConfig import moe.kyokobot.koe.KoeClient import moe.kyokobot.koe.KoeEventAdapter import moe.kyokobot.koe.VoiceConnection @@ -46,6 +47,7 @@ import java.util.concurrent.TimeUnit class SocketContext internal constructor( val audioPlayerManager: AudioPlayerManager, + val serverConfig: ServerConfig, private var session: WebSocketSession, private val socketServer: SocketServer, val userId: String, @@ -90,7 +92,7 @@ class SocketContext internal constructor( } internal fun getPlayer(guildId: String) = players.computeIfAbsent(guildId) { - Player(this, guildId, audioPlayerManager) + Player(this, guildId, audioPlayerManager, serverConfig) } internal fun getPlayers(): Map { diff --git a/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt b/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt index f93309258..9c5bddfce 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/SocketServer.kt @@ -89,6 +89,7 @@ class SocketServer( contextMap[session.id] = SocketContext( audioPlayerManager, + serverConfig, session, this, userId, diff --git a/LavalinkServer/src/main/java/lavalink/server/player/Player.java b/LavalinkServer/src/main/java/lavalink/server/player/Player.java index 8caf16641..e2ae73973 100644 --- a/LavalinkServer/src/main/java/lavalink/server/player/Player.java +++ b/LavalinkServer/src/main/java/lavalink/server/player/Player.java @@ -33,6 +33,7 @@ import io.netty.buffer.ByteBuf; import lavalink.server.io.SocketContext; import lavalink.server.io.SocketServer; +import lavalink.server.config.ServerConfig; import moe.kyokobot.koe.VoiceConnection; import moe.kyokobot.koe.media.OpusAudioFrameProvider; import org.json.JSONObject; @@ -49,6 +50,7 @@ public class Player extends AudioEventAdapter { private final SocketContext socketContext; private final String guildId; + private final ServerConfig serverConfig; private final AudioPlayer player; private final AudioLossCounter audioLossCounter = new AudioLossCounter(); private AudioFrame lastFrame = null; @@ -56,9 +58,10 @@ public class Player extends AudioEventAdapter { private final EqualizerFactory equalizerFactory = new EqualizerFactory(); private boolean isEqualizerApplied = false; - public Player(SocketContext socketContext, String guildId, AudioPlayerManager audioPlayerManager) { + public Player(SocketContext socketContext, String guildId, AudioPlayerManager audioPlayerManager, ServerConfig serverConfig) { this.socketContext = socketContext; this.guildId = guildId; + this.serverConfig = serverConfig; this.player = audioPlayerManager.createPlayer(); this.player.addListener(this); this.player.addListener(new EventEmitter(audioPlayerManager, this)); @@ -148,6 +151,10 @@ public AudioLossCounter getAudioLossCounter() { return audioLossCounter; } + private int getInterval() { + return serverConfig.getPlayerUpdateInterval(); + } + public boolean isPlaying() { return player.getPlayingTrack() != null && !player.isPaused(); } @@ -164,7 +171,7 @@ public void onTrackStart(AudioPlayer player, AudioTrack track) { if (socketContext.getSessionPaused()) return; SocketServer.Companion.sendPlayerUpdate(socketContext, this); - }, 0, 5, TimeUnit.SECONDS); + }, 0, this.getInterval(), TimeUnit.SECONDS); } } From 62d529c69d4e034223fe3377b906ce6dc6d62ae2 Mon Sep 17 00:00:00 2001 From: TheEssem Date: Mon, 6 Jul 2020 10:58:35 -0500 Subject: [PATCH 38/56] Set default value --- .../src/main/java/lavalink/server/config/ServerConfig.kt | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/config/ServerConfig.kt b/LavalinkServer/src/main/java/lavalink/server/config/ServerConfig.kt index 84c043d19..28f27ffe9 100644 --- a/LavalinkServer/src/main/java/lavalink/server/config/ServerConfig.kt +++ b/LavalinkServer/src/main/java/lavalink/server/config/ServerConfig.kt @@ -33,7 +33,7 @@ class ServerConfig { var sentryDsn = "" var bufferDurationMs: Int? = null var youtubePlaylistLoadLimit: Int? = null - var playerUpdateInterval: Int? = null + var playerUpdateInterval: Int? = 5 var isGcWarnings = true var isYoutubeSearchEnabled = true var isSoundcloudSearchEnabled = true From 017dd3263dc143359cfec5aeb865ce6621529e4d Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Fri, 10 Jul 2020 21:31:30 +0200 Subject: [PATCH 39/56] Update Koe to stable 1.0.0 --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index 3674fcf80..7303ca83c 100644 --- a/build.gradle +++ b/build.gradle @@ -57,7 +57,7 @@ subprojects { lavaplayerIpRotatorVersion = '0.2.3' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' - koeVersion = '1.0.0-beta3' + koeVersion = '1.0.0' springBootVersion = "${springBootVersion}" springWebSocketVersion = '5.1.9.RELEASE' From bad180a7cee1c9cfd50cf01db13000fd6cbccb4c Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Mon, 13 Jul 2020 11:13:55 +0200 Subject: [PATCH 40/56] Add `playerUpdateInterval` comment --- LavalinkServer/application.yml.example | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/application.yml.example b/LavalinkServer/application.yml.example index 2838b310e..1cb643c6f 100644 --- a/LavalinkServer/application.yml.example +++ b/LavalinkServer/application.yml.example @@ -15,7 +15,7 @@ lavalink: local: false bufferDurationMs: 400 youtubePlaylistLoadLimit: 6 # Number of pages at 100 each - playerUpdateInterval: 5 + playerUpdateInterval: 5 # How frequently to send player updates to clients, in seconds youtubeSearchEnabled: true soundcloudSearchEnabled: true gc-warnings: true From c581b47420733c6989e2c81f92100963c00eadc6 Mon Sep 17 00:00:00 2001 From: Sangoon_Is_Noob Date: Tue, 21 Jul 2020 18:19:02 +0900 Subject: [PATCH 41/56] Fixed #332 (JSONObject["address"] not found.) --- .../main/java/lavalink/server/io/RoutePlannerRestHandler.kt | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/LavalinkServer/src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt b/LavalinkServer/src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt index b86ac1998..00f6789b6 100644 --- a/LavalinkServer/src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt +++ b/LavalinkServer/src/main/java/lavalink/server/io/RoutePlannerRestHandler.kt @@ -39,10 +39,11 @@ class RoutePlannerRestHandler(private val routePlanner: AbstractRoutePlanner?) { * Removes a single address from the addresses which are currently marked as failing */ @PostMapping("/routeplanner/free/address") - fun freeSingleAddress(request: HttpServletRequest, @RequestBody requestBody: JSONObject): ResponseEntity { + fun freeSingleAddress(request: HttpServletRequest, @RequestBody requestBody: String): ResponseEntity { routePlanner ?: throw RoutePlannerDisabledException() try { - val address = InetAddress.getByName(requestBody.getString("address")) + val jsonObject = JSONObject(requestBody) + val address = InetAddress.getByName(jsonObject.getString("address")) routePlanner.freeAddress(address) return ResponseEntity.noContent().build() } catch (exception: UnknownHostException) { From 320e71bd12ce2fdfd3045796b8ad01eeaab59dd6 Mon Sep 17 00:00:00 2001 From: Emad Abdullah Date: Tue, 4 Aug 2020 20:54:16 +0300 Subject: [PATCH 42/56] Update Koe --- build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build.gradle b/build.gradle index 7303ca83c..9b9c3d144 100644 --- a/build.gradle +++ b/build.gradle @@ -57,7 +57,7 @@ subprojects { lavaplayerIpRotatorVersion = '0.2.3' jdaNasVersion = '1.1.0' jappVersion = '1.3.2-MINN' - koeVersion = '1.0.0' + koeVersion = '1.0.1' springBootVersion = "${springBootVersion}" springWebSocketVersion = '5.1.9.RELEASE' From f7c43981cae46bf00e7632cb350e1331034d4035 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Tue, 22 Sep 2020 15:44:48 +0200 Subject: [PATCH 43/56] Use Devoxin lavaplayer fork --- LavalinkServer/build.gradle | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index 783f666ed..81e133cb2 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -39,7 +39,8 @@ dependencies { exclude group: "org.slf4j", module: "slf4j-api" } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion - compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion + //compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion + compile group: 'com.github.devoxin', name: 'lavaplayer', version: '70f53b91' compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion From bd26992226cc2747faf310f4fe6b869d34d7fd65 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Wed, 23 Sep 2020 22:28:24 +0200 Subject: [PATCH 44/56] Update lavaplayer --- LavalinkServer/build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index 81e133cb2..bce5736e6 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -40,7 +40,7 @@ dependencies { } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion //compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion - compile group: 'com.github.devoxin', name: 'lavaplayer', version: '70f53b91' + compile group: 'com.github.devoxin', name: 'lavaplayer', version: '1.3.52' compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion From 0a2240a5a70f95548addd047b914ebf6a8b8c390 Mon Sep 17 00:00:00 2001 From: nitsuga5124 Date: Thu, 13 Aug 2020 15:27:40 +0200 Subject: [PATCH 45/56] Added the Rust lavalink library. modified: README.md --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index ff98fa40e..948fc4476 100644 --- a/README.md +++ b/README.md @@ -89,6 +89,7 @@ users about the compatibility of their clients to the Lavalink server. * [Magma](https://github.com/initzx/magma/) (discord.py, Python) * [lavapotion](https://github.com/SamOphis/lavapotion) (Elixir) * [WaveLink](https://github.com/EvieePy/Wavelink)(discord.py, Python) +* [Lavalink-rs](https://gitlab.com/nitsuga5124/lavalink-rs/)(Async Libraries, Rust) * Or [create your own](https://github.com/Frederikam/Lavalink/blob/master/IMPLEMENTATION.md) ## Server configuration From 3b1e9a6ed65ac51ec1d755ca88fd360172fc87a0 Mon Sep 17 00:00:00 2001 From: nitsuga5124 Date: Thu, 13 Aug 2020 15:40:13 +0200 Subject: [PATCH 46/56] Spacing modified: README.md --- README.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index 948fc4476..985d202b1 100644 --- a/README.md +++ b/README.md @@ -88,8 +88,8 @@ users about the compatibility of their clients to the Lavalink server. * [gavalink](https://github.com/foxbot/gavalink) (Go) * [Magma](https://github.com/initzx/magma/) (discord.py, Python) * [lavapotion](https://github.com/SamOphis/lavapotion) (Elixir) -* [WaveLink](https://github.com/EvieePy/Wavelink)(discord.py, Python) -* [Lavalink-rs](https://gitlab.com/nitsuga5124/lavalink-rs/)(Async Libraries, Rust) +* [WaveLink](https://github.com/EvieePy/Wavelink) (discord.py, Python) +* [Lavalink-rs](https://gitlab.com/nitsuga5124/lavalink-rs/) (Async Libraries, Rust) * Or [create your own](https://github.com/Frederikam/Lavalink/blob/master/IMPLEMENTATION.md) ## Server configuration From b080f31f0f93d76200c4b6d62619e1e15de101f0 Mon Sep 17 00:00:00 2001 From: God <62307220+Echo-3-1@users.noreply.github.com> Date: Thu, 3 Sep 2020 15:03:33 +0530 Subject: [PATCH 47/56] Update README.md --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 985d202b1..ba285b949 100644 --- a/README.md +++ b/README.md @@ -90,6 +90,7 @@ users about the compatibility of their clients to the Lavalink server. * [lavapotion](https://github.com/SamOphis/lavapotion) (Elixir) * [WaveLink](https://github.com/EvieePy/Wavelink) (discord.py, Python) * [Lavalink-rs](https://gitlab.com/nitsuga5124/lavalink-rs/) (Async Libraries, Rust) +* [EvoLava](https://github.com/EvolveJS/EvoLava) ([EvolveJS](https://github.com/EvolveJS/EvolveJS), Typescript, Javascript) * Or [create your own](https://github.com/Frederikam/Lavalink/blob/master/IMPLEMENTATION.md) ## Server configuration From 499f2706fbe5d07689b1cf3e52015499d0c27567 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Thu, 3 Sep 2020 12:46:22 +0200 Subject: [PATCH 48/56] Move lib in line with other JS libs --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index ba285b949..116b2f097 100644 --- a/README.md +++ b/README.md @@ -80,6 +80,7 @@ users about the compatibility of their clients to the Lavalink server. * [Shoukaku](https://github.com/Deivu/Shoukaku) ([discord.js](https://github.com/discordjs/discord.js), JavaScript) * [Lavacord](https://github.com/lavacord/lavacord) (JavaScript) * [LavaJS](https://github.com/Projects-Me/LavaJS) ([discord.js](https://github.com/discordjs/discord.js), JavaScript/TypeScript) +* [EvoLava](https://github.com/EvolveJS/EvoLava) ([EvolveJS](https://github.com/EvolveJS/EvolveJS), Javascript/Typescript) * [SharpLink](https://github.com/Devoxin/SharpLink) ([Discord.Net](https://github.com/RogueException/Discord.Net), .NET) * [Victoria](https://github.com/Yucked/Victoria) ([Discord.NET](https://github.com/RogueException/Discord.Net), .NET) * [Lavalink.NET](https://github.com/Dev-Yukine/Lavalink.NET) (.NET) @@ -90,7 +91,6 @@ users about the compatibility of their clients to the Lavalink server. * [lavapotion](https://github.com/SamOphis/lavapotion) (Elixir) * [WaveLink](https://github.com/EvieePy/Wavelink) (discord.py, Python) * [Lavalink-rs](https://gitlab.com/nitsuga5124/lavalink-rs/) (Async Libraries, Rust) -* [EvoLava](https://github.com/EvolveJS/EvoLava) ([EvolveJS](https://github.com/EvolveJS/EvolveJS), Typescript, Javascript) * Or [create your own](https://github.com/Frederikam/Lavalink/blob/master/IMPLEMENTATION.md) ## Server configuration From 0870bfdb0420fcb0ebcd8a6492dbc75f8ed5505b Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Tue, 29 Sep 2020 12:31:06 +0200 Subject: [PATCH 49/56] Rebase --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 16c641bc7..4c7658b87 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -3,6 +3,8 @@ Each release usually includes various fixes and improvements. The most noteworthy of these, as well as any features and breaking changes, are listed here. +## v3.3.1.2 +* Update lavaplayer to [@Devoxin](https://github.com/Devoxin)'s' fork ## v3.3.1.1 * Updated Lavaplayer to `1.3.50`. This notably fixes YouTube search. From a4710455fab4c5c200ee54e8affd2f344972945e Mon Sep 17 00:00:00 2001 From: Fabricio Winter Date: Tue, 29 Sep 2020 07:22:39 -0300 Subject: [PATCH 50/56] Update lavaplayer --- CHANGELOG.md | 3 +++ LavalinkServer/build.gradle | 2 +- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4c7658b87..e4a12c468 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -3,6 +3,9 @@ Each release usually includes various fixes and improvements. The most noteworthy of these, as well as any features and breaking changes, are listed here. +## v3.3.1.3 +* Update lavaplayer to `1.3.53` from devoxin's fork. + ## v3.3.1.2 * Update lavaplayer to [@Devoxin](https://github.com/Devoxin)'s' fork diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index bce5736e6..149dfbd3b 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -40,7 +40,7 @@ dependencies { } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion //compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion - compile group: 'com.github.devoxin', name: 'lavaplayer', version: '1.3.52' + compile group: 'com.github.devoxin', name: 'lavaplayer', version: 'a63c541dc5' compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion From 385de6a7eb702f0c5a6ba6072fbedd121d6e8bd8 Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Tue, 29 Sep 2020 13:35:49 +0200 Subject: [PATCH 51/56] Update Gradle --- build.gradle | 6 --- gradle/wrapper/gradle-wrapper.jar | Bin 56177 -> 58910 bytes gradle/wrapper/gradle-wrapper.properties | 2 +- gradlew | 53 ++++++++++++++--------- gradlew.bat | 22 +++++++++- 5 files changed, 55 insertions(+), 28 deletions(-) diff --git a/build.gradle b/build.gradle index 9b9c3d144..351b697c9 100644 --- a/build.gradle +++ b/build.gradle @@ -76,9 +76,3 @@ subprojects { ext { moduleName = 'Lavalink-Parent' } - -task wrapper(type: Wrapper) { - gradleVersion = '4.10.2' - //noinspection UnnecessaryQualifiedReference - distributionType = Wrapper.DistributionType.ALL -} diff --git a/gradle/wrapper/gradle-wrapper.jar b/gradle/wrapper/gradle-wrapper.jar index 29953ea141f55e3b8fc691d31b5ca8816d89fa87..62d4c053550b91381bbd28b1afc82d634bf73a8a 100644 GIT binary patch delta 50641 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�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%0DQAk`0X;4B`zXE~vg6f#?WVEZo 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 diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index d76b502e2..622ab64a3 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,5 +1,5 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-4.10.2-all.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-6.5-bin.zip zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists diff --git a/gradlew b/gradlew index cccdd3d51..fbd7c5158 100755 --- a/gradlew +++ b/gradlew @@ -1,5 +1,21 @@ #!/usr/bin/env sh +# +# Copyright 2015 the original author or authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + ############################################################################## ## ## Gradle start up script for UN*X @@ -28,7 +44,7 @@ APP_NAME="Gradle" APP_BASE_NAME=`basename "$0"` # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. -DEFAULT_JVM_OPTS="" +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' # Use the maximum available, or set MAX_FD != -1 to use that value. MAX_FD="maximum" @@ -66,6 +82,7 @@ esac CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + # Determine the Java command to use to start the JVM. if [ -n "$JAVA_HOME" ] ; then if [ -x "$JAVA_HOME/jre/sh/java" ] ; then @@ -109,10 +126,11 @@ if $darwin; then GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" fi -# For Cygwin, switch paths to Windows format before running java -if $cygwin ; then +# For Cygwin or MSYS, switch paths to Windows format before running java +if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then APP_HOME=`cygpath --path --mixed "$APP_HOME"` CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` + JAVACMD=`cygpath --unix "$JAVACMD"` # We build the pattern for arguments to be converted via cygpath @@ -138,19 +156,19 @@ if $cygwin ; then else eval `echo args$i`="\"$arg\"" fi - i=$((i+1)) + i=`expr $i + 1` done case $i in - (0) set -- ;; - (1) set -- "$args0" ;; - (2) set -- "$args0" "$args1" ;; - (3) set -- "$args0" "$args1" "$args2" ;; - (4) set -- "$args0" "$args1" "$args2" "$args3" ;; - (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; - (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; - (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; - (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; - (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; + 0) set -- ;; + 1) set -- "$args0" ;; + 2) set -- "$args0" "$args1" ;; + 3) set -- "$args0" "$args1" "$args2" ;; + 4) set -- "$args0" "$args1" "$args2" "$args3" ;; + 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; + 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; + 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; + 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; + 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; esac fi @@ -159,14 +177,9 @@ save () { for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done echo " " } -APP_ARGS=$(save "$@") +APP_ARGS=`save "$@"` # Collect all arguments for the java command, following the shell quoting and substitution rules eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" -# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong -if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then - cd "$(dirname "$0")" -fi - exec "$JAVACMD" "$@" diff --git a/gradlew.bat b/gradlew.bat index e95643d6a..a9f778a7a 100644 --- a/gradlew.bat +++ b/gradlew.bat @@ -1,3 +1,19 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem + @if "%DEBUG%" == "" @echo off @rem ########################################################################## @rem @@ -13,8 +29,11 @@ if "%DIRNAME%" == "" set DIRNAME=. set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. -set DEFAULT_JVM_OPTS= +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" @rem Find java.exe if defined JAVA_HOME goto findJavaFromJavaHome @@ -65,6 +84,7 @@ set CMD_LINE_ARGS=%* set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + @rem Execute Gradle "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% From ed24d4f10584edfc6c63bd894c876931d987e84f Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sun, 4 Oct 2020 16:33:02 +0200 Subject: [PATCH 52/56] Set Sentry environment from config --- LavalinkServer/application.yml.example | 1 + .../server/config/SentryConfigProperties.java | 34 ------------------- .../server/config/SentryConfigProperties.kt | 16 +++++++++ .../server/config/SentryConfiguration.java | 5 +-- 4 files changed, 20 insertions(+), 36 deletions(-) delete mode 100644 LavalinkServer/src/main/java/lavalink/server/config/SentryConfigProperties.java create mode 100644 LavalinkServer/src/main/java/lavalink/server/config/SentryConfigProperties.kt diff --git a/LavalinkServer/application.yml.example b/LavalinkServer/application.yml.example index 1cb643c6f..868d1e836 100644 --- a/LavalinkServer/application.yml.example +++ b/LavalinkServer/application.yml.example @@ -33,6 +33,7 @@ metrics: sentry: dsn: "" + environment: "" # tags: # some_key: some_value # another_key: another_value diff --git a/LavalinkServer/src/main/java/lavalink/server/config/SentryConfigProperties.java b/LavalinkServer/src/main/java/lavalink/server/config/SentryConfigProperties.java deleted file mode 100644 index eca63fa28..000000000 --- a/LavalinkServer/src/main/java/lavalink/server/config/SentryConfigProperties.java +++ /dev/null @@ -1,34 +0,0 @@ -package lavalink.server.config; - -import org.springframework.boot.context.properties.ConfigurationProperties; -import org.springframework.stereotype.Component; - -import java.util.HashMap; -import java.util.Map; - -/** - * Created by napster on 20.05.18. - */ -@Component -@ConfigurationProperties(prefix = "sentry") -public class SentryConfigProperties { - - private String dsn = ""; - private Map tags = new HashMap<>(); - - public String getDsn() { - return dsn; - } - - public void setDsn(String dsn) { - this.dsn = dsn; - } - - public Map getTags() { - return tags; - } - - public void setTags(Map tags) { - this.tags = tags; - } -} diff --git a/LavalinkServer/src/main/java/lavalink/server/config/SentryConfigProperties.kt b/LavalinkServer/src/main/java/lavalink/server/config/SentryConfigProperties.kt new file mode 100644 index 000000000..2aec8e6ec --- /dev/null +++ b/LavalinkServer/src/main/java/lavalink/server/config/SentryConfigProperties.kt @@ -0,0 +1,16 @@ +package lavalink.server.config + +import org.springframework.boot.context.properties.ConfigurationProperties +import org.springframework.stereotype.Component +import java.util.* + +/** + * Created by napster on 20.05.18. + */ +@Component +@ConfigurationProperties(prefix = "sentry") +class SentryConfigProperties { + var dsn = "" + var environment = "" + var tags: Map = HashMap() +} \ No newline at end of file diff --git a/LavalinkServer/src/main/java/lavalink/server/config/SentryConfiguration.java b/LavalinkServer/src/main/java/lavalink/server/config/SentryConfiguration.java index f0fa9b34d..b4e420d94 100644 --- a/LavalinkServer/src/main/java/lavalink/server/config/SentryConfiguration.java +++ b/LavalinkServer/src/main/java/lavalink/server/config/SentryConfiguration.java @@ -36,7 +36,7 @@ public SentryConfiguration(ServerConfig serverConfig, SentryConfigProperties sen } if (dsn != null && !dsn.isEmpty()) { - turnOn(dsn, sentryConfig.getTags()); + turnOn(dsn, sentryConfig.getTags(), sentryConfig.getEnvironment()); if (warnDeprecatedDsnConfig) { log.warn("Please update the location of the sentry dsn in lavalinks config file / your environment " + "vars, it is now located under 'sentry.dsn' instead of 'lavalink.server.sentryDsn'."); @@ -47,9 +47,10 @@ public SentryConfiguration(ServerConfig serverConfig, SentryConfigProperties sen } - public void turnOn(String dsn, Map tags) { + public void turnOn(String dsn, Map tags, String environment) { log.info("Turning on sentry"); SentryClient sentryClient = Sentry.init(dsn); + if (!environment.isBlank()) sentryClient.setEnvironment(environment); if (tags != null) { tags.forEach(sentryClient::addTag); From d55f4ee03eb24e08ec19c2ffbbccc10139464632 Mon Sep 17 00:00:00 2001 From: Devoxin <15076404+Devoxin@users.noreply.github.com> Date: Sun, 4 Oct 2020 15:51:40 +0100 Subject: [PATCH 53/56] Update LP --- LavalinkServer/build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index 149dfbd3b..d2e90856b 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -40,7 +40,7 @@ dependencies { } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion //compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion - compile group: 'com.github.devoxin', name: 'lavaplayer', version: 'a63c541dc5' + compile group: 'com.github.devoxin', name: 'lavaplayer', version: '1.3.54.3' compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion From bdf8446e38ac7c0738bb1d0b536bb8b8ec5b2da6 Mon Sep 17 00:00:00 2001 From: Devoxin <15076404+Devoxin@users.noreply.github.com> Date: Wed, 14 Oct 2020 17:43:15 +0100 Subject: [PATCH 54/56] Update build.gradle --- LavalinkServer/build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index d2e90856b..ab4268f29 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -40,7 +40,7 @@ dependencies { } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion //compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion - compile group: 'com.github.devoxin', name: 'lavaplayer', version: '1.3.54.3' + compile group: 'com.github.devoxin', name: 'lavaplayer', version: '1.3.58' compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion From 43726901f37d6c6e220cb211390ea044fbe4648b Mon Sep 17 00:00:00 2001 From: caneleex Date: Mon, 19 Oct 2020 18:48:51 +0200 Subject: [PATCH 55/56] bump lavaplayer --- LavalinkServer/build.gradle | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/LavalinkServer/build.gradle b/LavalinkServer/build.gradle index ab4268f29..5138aefc0 100644 --- a/LavalinkServer/build.gradle +++ b/LavalinkServer/build.gradle @@ -40,7 +40,7 @@ dependencies { } compile group: 'moe.kyokobot.koe', name: 'ext-udpqueue', version: koeVersion //compile group: 'com.sedmelluq', name: 'lavaplayer', version: lavaplayerVersion - compile group: 'com.github.devoxin', name: 'lavaplayer', version: '1.3.58' + compile group: 'com.github.devoxin', name: 'lavaplayer', version: '1.3.59' compile group: 'com.sedmelluq', name: 'lavaplayer-ext-youtube-rotator', version: lavaplayerIpRotatorVersion compile group: 'org.jetbrains.kotlin', name: 'kotlin-reflect', version: kotlinVersion From 187c5c4a7f469b7e67e67cb4ca9b7b421eac3b0e Mon Sep 17 00:00:00 2001 From: "Frederik Ar. Mikkelsen" Date: Sat, 24 Oct 2020 11:11:59 +0200 Subject: [PATCH 56/56] Release v3.3.2 --- CHANGELOG.md | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index e4a12c468..279df442b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -3,6 +3,24 @@ Each release usually includes various fixes and improvements. The most noteworthy of these, as well as any features and breaking changes, are listed here. +## v3.3.2 +* Replaced Magma with Koe. +* Finally implemented `stopTime` for `play` op. +* Added `playerUpdateInterval` config option. +* Added `environment` to Sentry config. +* Fixed #332 +* Updated IP rotator. +* Update lavaplayer to `1.3.59` from devoxin's fork. +* Added a Testbot for development. + +Contributors: +[@Frederikam](https://github.com/Frederikam), +[@Thewsomeguy](https://github.com/Thewsomeguy), +[@Neuheit](https://github.com/Neuheit), +[@Sangoon_Is_Noob](https://github.com/Sangoon_Is_Noob), +[@TheEssem](https://github.com/Essem), and +[@Devoxin](https://github.com/Devoxin) + ## v3.3.1.3 * Update lavaplayer to `1.3.53` from devoxin's fork.