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

com.jauntsdn.rsocket.internal.ClientServerInputMultiplexer Maven / Gradle / Ivy

There is a newer version: 0.9.8
Show newest version
/*
 * Copyright 2015-2018 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
 *
 *     http://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.
 */

package com.jauntsdn.rsocket.internal;

import com.jauntsdn.rsocket.Closeable;
import com.jauntsdn.rsocket.DuplexConnection;
import com.jauntsdn.rsocket.frame.FrameHeaderFlyweight;
import com.jauntsdn.rsocket.frame.FrameUtil;
import io.netty.buffer.ByteBuf;
import org.reactivestreams.Publisher;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
import reactor.core.publisher.MonoProcessor;
import reactor.core.scheduler.Scheduler;

/**
 * {@link DuplexConnection#receive()} is a single stream on which the following type of frames
 * arrive:
 *
 * 
    *
  • Frames for streams initiated by the initiator of the connection (client). *
  • Frames for streams initiated by the acceptor of the connection (server). *
* *

The only way to differentiate these two frames is determining whether the stream Id is odd or * even. Even IDs are for the streams initiated by server and odds are for streams initiated by the * client. */ public class ClientServerInputMultiplexer implements Closeable { private static final Logger LOGGER = LoggerFactory.getLogger("com.jauntsdn.rsocket.FrameLogger"); private final DuplexConnection setupConnection; private final DuplexConnection serverConnection; private final DuplexConnection clientConnection; private final DuplexConnection source; private final DuplexConnection clientServerConnection; public ClientServerInputMultiplexer(DuplexConnection source, boolean isClient) { this.source = source; final MonoProcessor> setup = MonoProcessor.create(); final MonoProcessor> server = MonoProcessor.create(); final MonoProcessor> client = MonoProcessor.create(); setupConnection = new InternalDuplexConnection(source, setup); serverConnection = new InternalDuplexConnection(source, server); clientConnection = new InternalDuplexConnection(source, client); clientServerConnection = new InternalDuplexConnection(source, client, server); source .receive() .groupBy( frame -> { int streamId = FrameHeaderFlyweight.streamId(frame); final ConnectionType type; if (streamId == 0) { switch (FrameHeaderFlyweight.frameType(frame)) { case SETUP: case RESUME: case RESUME_OK: type = ConnectionType.SETUP; break; case LEASE: case KEEPALIVE: case ERROR: type = isClient ? ConnectionType.CLIENT : ConnectionType.SERVER; break; default: type = isClient ? ConnectionType.SERVER : ConnectionType.CLIENT; } } else if ((streamId & 0b1) == 0) { type = ConnectionType.SERVER; } else { type = ConnectionType.CLIENT; } return type; }) .subscribe( group -> { switch (group.key()) { case SETUP: setup.onNext(group); break; case SERVER: server.onNext(group); break; case CLIENT: client.onNext(group); break; } }, t -> { if (LOGGER.isDebugEnabled()) { LOGGER.error("Error receiving frame:", t); } dispose(); }); } public DuplexConnection asClientServerConnection() { return clientServerConnection; } public DuplexConnection asServerConnection() { return serverConnection; } public DuplexConnection asClientConnection() { return clientConnection; } public DuplexConnection asSetupConnection() { return setupConnection; } @Override public void dispose() { source.dispose(); } @Override public boolean isDisposed() { return source.isDisposed(); } @Override public Mono onClose() { return source.onClose(); } private enum ConnectionType { SETUP, CLIENT, SERVER, SOURCE } private static class InternalDuplexConnection implements DuplexConnection { private final DuplexConnection source; private final MonoProcessor>[] processors; private final boolean debugEnabled; public InternalDuplexConnection( DuplexConnection source, MonoProcessor>... processors) { this.source = source; this.processors = processors; this.debugEnabled = LOGGER.isDebugEnabled(); } @Override public Mono send(Publisher frame) { if (debugEnabled) { frame = Flux.from(frame).doOnNext(f -> LOGGER.debug("sending -> " + FrameUtil.toString(f))); } return source.send(frame); } @Override public Mono sendOne(ByteBuf frame) { if (debugEnabled) { LOGGER.debug("sending -> " + FrameUtil.toString(frame)); } return source.sendOne(frame); } @Override public Flux receive() { return Flux.fromArray(processors) .flatMap( p -> p.flatMapMany( f -> { if (debugEnabled) { return f.doOnNext( frame -> LOGGER.debug("receiving -> " + FrameUtil.toString(frame))); } else { return f; } })); } @Override public void dispose() { source.dispose(); } @Override public boolean isDisposed() { return source.isDisposed(); } @Override public Mono onClose() { return source.onClose(); } @Override public double availability() { return source.availability(); } @Override public Scheduler scheduler() { return source.scheduler(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy