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

org.springframework.integration.channel.FluxMessageChannel Maven / Gradle / Ivy

There is a newer version: 6.3.3
Show newest version
/*
 * Copyright 2002-2019 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.
 */

package org.springframework.integration.channel;

import org.reactivestreams.Publisher;
import org.reactivestreams.Subscriber;

import org.springframework.messaging.Message;
import org.springframework.util.Assert;

import reactor.core.Disposable;
import reactor.core.Disposables;
import reactor.core.publisher.EmitterProcessor;
import reactor.core.publisher.Flux;
import reactor.core.publisher.FluxSink;
import reactor.core.publisher.ReplayProcessor;
import reactor.core.scheduler.Schedulers;

/**
 * The {@link AbstractMessageChannel} implementation for the
 * Reactive Streams {@link Publisher} based on the Project Reactor {@link Flux}.
 *
 * @author Artem Bilan
 * @author Gary Russell
 * @author Sergei Egorov
 *
 * @since 5.0
 */
public class FluxMessageChannel extends AbstractMessageChannel
		implements Publisher>, ReactiveStreamsSubscribableChannel {

	private final EmitterProcessor> processor;

	private final FluxSink> sink;

	private final ReplayProcessor subscribedSignal = ReplayProcessor.create(1);

	private final Disposable.Composite upstreamSubscriptions = Disposables.composite();

	public FluxMessageChannel() {
		this.processor = EmitterProcessor.create(1, false);
		this.sink = this.processor.sink(FluxSink.OverflowStrategy.BUFFER);
	}

	@Override
	protected boolean doSend(Message message, long timeout) {
		Assert.state(this.processor.hasDownstreams(),
				() -> "The [" + this + "] doesn't have subscribers to accept messages");
		this.sink.next(message);
		return true;
	}

	@Override
	public void subscribe(Subscriber> subscriber) {
		this.processor
				.doFinally((s) -> this.subscribedSignal.onNext(this.processor.hasDownstreams()))
				.subscribe(subscriber);
		this.subscribedSignal.onNext(this.processor.hasDownstreams());
	}

	@Override
	public void subscribeTo(Publisher> publisher) {
		this.upstreamSubscriptions.add(
				Flux.from(publisher)
						.delaySubscription(this.subscribedSignal.filter(Boolean::booleanValue).next())
						.publishOn(Schedulers.boundedElastic())
						.doOnNext((message) -> {
							try {
								send(message);
							}
							catch (Exception e) {
								logger.warn("Error during processing event: " + message, e);
							}
						})
						.subscribe());
	}

	@Override
	public void destroy() {
		this.subscribedSignal.onNext(false);
		this.upstreamSubscriptions.dispose();
		this.processor.onComplete();
		super.destroy();
	}

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy