net.oneandone.reactive.sse.servlet.ServletSseSubscriber Maven / Gradle / Ivy
/*
* Copyright 1&1 Internet AG, https://github.com/1and1/
*
* 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 net.oneandone.reactive.sse.servlet;
import java.time.Duration;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicReference;
import java.util.logging.Logger;
import javax.servlet.ServletOutputStream;
import net.oneandone.reactive.sse.ServerSentEvent;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
/**
* Maps Server-sent events (reactive) stream into a servlet output streamr
*
*/
public class ServletSseSubscriber implements Subscriber {
private static final Logger LOG = Logger.getLogger(ServletSseSubscriber.class.getName());
private final AtomicBoolean isOpen = new AtomicBoolean(true);
private final AtomicReference subscriptionRef = new AtomicReference<>(new IllegalStateSubscription());
private final SseWriteableChannel channel;
/**
* @param out the servlet output stream
* @param executor the scheduled executor to emit keep alive messages
*/
public ServletSseSubscriber(ServletOutputStream out, ScheduledExecutorService executor) {
this(out, executor, Duration.ofSeconds(25));
}
/**
* @param out the servlet output stream
* @param executor the scheduled executor to emit keep alive messages
* @param keepAlivePeriod the keep alive period
*/
public ServletSseSubscriber(ServletOutputStream out,ScheduledExecutorService executor, Duration keepAlivePeriod) {
this.channel = new SseWriteableChannel(out, error -> onError(error), keepAlivePeriod, executor);
}
@Override
public void onSubscribe(Subscription subscription) {
subscriptionRef.set(subscription);
subscriptionRef.get().request(25); // request next 25
}
@Override
public void onNext(ServerSentEvent event) {
channel.writeEventAsync(event)
.thenAccept(written -> subscriptionRef.get().request(1));
}
@Override
public void onError(Throwable t) {
LOG.fine("error on source stream. stop streaming " + t.getMessage());
close();
}
@Override
public void onComplete() {
close();
}
private void close() {
if (isOpen.getAndSet(false)) {
subscriptionRef.get().cancel();
channel.close();
}
}
private static final class IllegalStateSubscription implements Subscription {
@Override
public void request(long n) {
throw new IllegalStateException();
}
@Override
public void cancel() {
throw new IllegalStateException();
}
}
}