All Downloads are FREE. Search and download functionalities are using the official Maven repository.

discord4j.voice.VoiceWebsocketHandler Maven / Gradle / Ivy

/*
 * This file is part of Discord4J.
 *
 * Discord4J is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * Discord4J is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with Discord4J. If not, see .
 */

package discord4j.voice;

import discord4j.common.close.CloseStatus;
import discord4j.common.close.DisconnectBehavior;
import discord4j.common.sinks.EmissionStrategy;
import discord4j.voice.retry.PartialDisconnectException;
import discord4j.voice.retry.VoiceGatewayException;
import io.netty.buffer.ByteBuf;
import io.netty.handler.codec.http.websocketx.CloseWebSocketFrame;
import io.netty.handler.codec.http.websocketx.TextWebSocketFrame;
import io.netty.handler.codec.http.websocketx.WebSocketFrame;
import reactor.core.publisher.Flux;
import reactor.core.publisher.FluxSink;
import reactor.core.publisher.Mono;
import reactor.core.publisher.Sinks;
import reactor.netty.http.websocket.WebsocketInbound;
import reactor.netty.http.websocket.WebsocketOutbound;
import reactor.util.Logger;
import reactor.util.Loggers;
import reactor.util.context.ContextView;
import reactor.util.function.Tuple2;

import java.time.Duration;

import static discord4j.common.LogUtil.format;
import static reactor.core.publisher.Sinks.EmitFailureHandler.FAIL_FAST;

/**
 * Represents a WebSocket handler specialized for Discord voice gateway operations.
 * 

* Capable of handling closing events that normally occur in its lifecycle. *

* This handler uses a {@link FluxSink} of {@link ByteBuf} to push inbound payloads and a {@link Flux} of * {@link ByteBuf} to pull outbound payloads. *

* The handler also provides methods to control the lifecycle, which perform operations on the current session. It is * required to use them to properly release important resources and complete the session. */ public class VoiceWebsocketHandler { private static final Logger log = Loggers.getLogger(VoiceWebsocketHandler.class); private final Sinks.Many inbound; private final Flux outbound; private final Sinks.One sessionClose; private final ContextView context; private final EmissionStrategy emissionStrategy; /** * Create a new handler with the given data pipelines. * * @param inbound the {@link Sinks.Many} of {@link ByteBuf} to process inbound payloads * @param outbound the {@link Flux} of {@link ByteBuf} to process outbound payloads * @param context the Reactor {@link ContextView} that owns this handler, to enrich logging */ public VoiceWebsocketHandler(Sinks.Many inbound, Flux outbound, ContextView context) { this.inbound = inbound; this.outbound = outbound; this.sessionClose = Sinks.one(); this.context = context; this.emissionStrategy = EmissionStrategy.park(Duration.ofNanos(10)); } /** * Handle an upgraded websocket connection, given by both {@link WebsocketInbound} and {@link WebsocketOutbound} to * manage a session until the remote closes or one of the local methods {@link #close()} or * {@link #error(Throwable)} methods are called. When that happens, a close procedure will take place and ultimately * emit a pair of {@link DisconnectBehavior} and remote {@link CloseStatus}, if present or "-1" if none is present. * * @param in the websocket inbound * @param out the websocket outbound * @return a {@link Mono} that upon subscription, manages a websocket session until it closes where a {@link Tuple2} * is emitted representing both the {@link DisconnectBehavior} that initiated the close procedure, and the inbound * {@link CloseStatus}. */ public Mono> handle(WebsocketInbound in, WebsocketOutbound out) { Mono outboundClose = sessionClose.asMono() .doOnNext(behavior -> log.debug(format(context, "Closing session with behavior: {}"), behavior)) .flatMap(behavior -> { switch (behavior.getAction()) { case RETRY_ABRUPTLY: case STOP_ABRUPTLY: return Mono.error(behavior.getCause() != null ? behavior.getCause() : new PartialDisconnectException(context)); case RETRY: case STOP: default: return Mono.just(CloseStatus.NORMAL_CLOSE); } }) .map(status -> new CloseWebSocketFrame(status.getCode(), status.getReason().orElse(null))); Mono inboundClose = in.receiveCloseStatus() .map(status -> new CloseStatus(status.code(), status.reasonText())) .doOnNext(status -> { log.debug(format(context, "Received close status: {}"), status); // TODO: discord uses code 4014 for both resumable and non-resumable disconnects // we optimistically issue a retry. might encounter a 4006 if invalid close(DisconnectBehavior.retryAbruptly(new VoiceGatewayException(context, "Inbound close status"))); }); Mono outboundEvents = out.sendObject(Flux.merge(outboundClose, outbound.map(TextWebSocketFrame::new))) .then(); in.withConnection(c -> c.onDispose(() -> log.debug(format(context, "Connection disposed")))); Mono inboundEvents = in.aggregateFrames() .receiveFrames() .map(WebSocketFrame::content) .doOnNext(this::emitInbound) .then(); return Mono.zip(outboundEvents, inboundEvents) .doOnError(this::error) .onErrorResume(t -> t.getCause() instanceof VoiceGatewayException, t -> Mono.empty()) .then(Mono.zip(sessionClose.asMono(), inboundClose.defaultIfEmpty(CloseStatus.ABNORMAL_CLOSE))); } private void emitInbound(ByteBuf value) { emissionStrategy.emitNext(inbound, value); } /** * Initiates a close sequence that will terminate this session and instruct consumers downstream that a reconnect * should take place afterwards. */ public void close() { close(DisconnectBehavior.retry(null)); } /** * Initiates a close sequence that will terminate this session and then execute a given {@link DisconnectBehavior}. * * @param behavior the {@link DisconnectBehavior} to follow after the close sequence starts */ public void close(DisconnectBehavior behavior) { sessionClose.emitValue(behavior, FAIL_FAST); } /** * Initiates a close sequence with the given error. The session will be terminated abruptly and then instruct * consumers downstream that a reconnect should take place afterwards. * * @param error the cause for this session termination */ public void error(Throwable error) { log.info(format(context, "Triggering error sequence: {}"), error.toString()); close(DisconnectBehavior.retryAbruptly(error)); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy