Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

recordServerConnectionInactive never invoked after WebSocket upgrade #3222

Closed
joebyneil opened this issue May 3, 2024 · 1 comment · Fixed by #3229
Closed

recordServerConnectionInactive never invoked after WebSocket upgrade #3222

joebyneil opened this issue May 3, 2024 · 1 comment · Fixed by #3229
Assignees
Labels
type/bug A general bug
Milestone

Comments

@joebyneil
Copy link

Hey 👋

We noticed that when using WebSockets with our own metrics recorder, the total connection count is accurate, but the active connection count seems to climb forever. It looks like our override of recordServerConnectionInactive is not invoked in these cases.

Expected Behavior

recordServerConnectionInactive should be invoked when a WebSocket based connection ends, in addition to recordServerConnectionClosed.

When no clients are connected, there should have been an equal amount of calls to recordServerConnectionActive and recordServerConnectionInactive.

Actual Behavior

Only recordServerConnectionClosed is recorded, causing any counts of Active connections to become inaccurate.

Steps to Reproduce

I have included a reproducer in the dropdown below:

package com.example.demo;

import org.junit.jupiter.api.Test;
import reactor.core.publisher.Mono;
import reactor.netty.http.HttpProtocol;
import reactor.netty.http.client.HttpClient;
import reactor.netty.http.client.WebsocketClientSpec;
import reactor.netty.http.server.HttpServer;
import reactor.netty.http.server.HttpServerMetricsRecorder;
import reactor.netty.http.server.WebsocketServerSpec;

import java.net.SocketAddress;
import java.time.Duration;
import java.util.concurrent.atomic.LongAdder;

import static org.assertj.core.api.Assertions.assertThat;

public class MyTest {
  @Test
  public void testWebSocketConnectionCounting() {
    var metrics = new MetricsRecorder();
    var websocketServerSpec = WebsocketServerSpec.builder().build();
    var server = HttpServer.create()
        .protocol(HttpProtocol.HTTP11)
        .metrics(true, () -> metrics)
        .handle((req, res) ->
            res.sendWebsocket((in, out) -> out.sendString(Mono.just("test")), websocketServerSpec))
        .bindNow();

    var webSocketClientSpec = WebsocketClientSpec.builder().build();
    HttpClient.create()
        .remoteAddress(server::address)
        .websocket(webSocketClientSpec)
        .handle((in, out) -> out.sendClose()).then().block();

    assertThat(metrics.totalConnectionsCounter.intValue()).isEqualTo(0);
    assertThat(metrics.activeConnectionCounter.intValue()).isEqualTo(0); // Throws
  }

  // HTTP works as expected
  @Test
  public void testWebSocketConnectionCountingHttp() {
    var metrics = new MetricsRecorder();
    var server = HttpServer.create()
        .protocol(HttpProtocol.HTTP11)
        .metrics(true, () -> metrics)
        .handle((req, res) -> res.send())
        .bindNow();

    HttpClient.create()
        .remoteAddress(server::address)
        .get().response().block();

    assertThat(metrics.totalConnectionsCounter.intValue()).isEqualTo(0);
    assertThat(metrics.activeConnectionCounter.intValue()).isEqualTo(0);
  }

  public class MetricsRecorder implements HttpServerMetricsRecorder {
    public final LongAdder activeConnectionCounter = new LongAdder();
    public final LongAdder totalConnectionsCounter = new LongAdder();

    public MetricsRecorder() {
    }

    @Override
    public void recordServerConnectionOpened(SocketAddress localAddress) {
      totalConnectionsCounter.increment();
    }

    @Override
    public void recordServerConnectionClosed(SocketAddress localAddress) {
      totalConnectionsCounter.decrement();
    }

    @Override
    public void recordServerConnectionActive(SocketAddress localAddress) {
      activeConnectionCounter.increment();
    }

    @Override
    public void recordServerConnectionInactive(SocketAddress localAddress) {
      activeConnectionCounter.decrement();
    }
    // Everything below this is noop for the example, but we do use them also
    @Override
    public void recordDataReceived(SocketAddress remoteAddress, long bytes) {
    }

    @Override
    public void recordDataSent(SocketAddress remoteAddress, long bytes) {
    }

    @Override
    public void incrementErrorsCount(SocketAddress remoteAddress) {
    }

    @Override
    public void recordTlsHandshakeTime(SocketAddress remoteAddress, Duration time, String status) {
    }

    @Override
    public void recordConnectTime(SocketAddress remoteAddress, Duration time, String status) {
    }

    @Override
    public void recordResolveAddressTime(SocketAddress remoteAddress, Duration time, String status) {
    }

    @Override
    public void recordDataReceivedTime(String uri, String method, Duration time) {
    }

    @Override
    public void recordDataSentTime(String uri, String method, String status, Duration time) {
    }

    @Override
    public void recordResponseTime(String uri, String method, String status, Duration time) {
    }

    @Override
    public void recordStreamOpened(SocketAddress localAddress) {
    }

    @Override
    public void recordStreamClosed(SocketAddress localAddress) {
    }

    @Override
    public void recordDataReceived(SocketAddress remoteAddress, String uri, long bytes) {
    }

    @Override
    public void recordDataSent(SocketAddress remoteAddress, String uri, long bytes) {
    }

    @Override
    public void incrementErrorsCount(SocketAddress remoteAddress, String uri) {
    }
  }
}

Possible Solution

We believe that this might be because the metrics handler is removed in WebSocketServerOperations.java.

Thinking that if we expect this not to record active/inactive for every WebSocket message, perhaps we should record that the connection is inactive at the completion of the upgrade. Otherwise, closing when the WebSocket connection ends would also work.

Your Environment

reactor-netty: 1.1.18

  • JVM version (java -version):
    openjdk version "17.0.6" 2023-01-17 LTS
    OpenJDK Runtime Environment Zulu17.40+19-CA (build 17.0.6+10-LTS)
    OpenJDK 64-Bit Server VM Zulu17.40+19-CA (build 17.0.6+10-LTS, mixed mode, sharing)
  • OS and version (eg. uname -a):
    Darwin Joebys-Laptop.local 23.4.0 Darwin Kernel Version 23.4.0: Fri Mar 15 00:10:42 PDT 2024; root:xnu-10063.101.17~1/RELEASE_ARM64_T6000 arm64
@joebyneil joebyneil added status/need-triage A new issue that still need to be evaluated as a whole type/bug A general bug labels May 3, 2024
@violetagg violetagg removed the status/need-triage A new issue that still need to be evaluated as a whole label May 5, 2024
@violetagg violetagg self-assigned this May 5, 2024
@violetagg violetagg added this to the 1.0.45 milestone May 5, 2024
@violetagg
Copy link
Member

@joebyneil Thanks for the report and the reproducible example. I confirm that this is an issue and it will be fixed for the next release.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
type/bug A general bug
Projects
None yet
2 participants