com.mongodb.async.client.AbstractSubscription Maven / Gradle / Ivy
/*
* Copyright 2015 MongoDB, Inc.
*
* 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.mongodb.async.client;
import java.util.List;
import java.util.concurrent.ConcurrentLinkedQueue;
abstract class AbstractSubscription implements Subscription {
private final Observer super TResult> observer;
/* protected by `this` */
private boolean requestedData;
private boolean isProcessing;
private long requested = 0;
private boolean isUnsubscribed = false;
private boolean isTerminated = false;
/* protected by `this` */
private final ConcurrentLinkedQueue resultsQueue = new ConcurrentLinkedQueue();
public AbstractSubscription(final Observer super TResult> observer) {
this.observer = observer;
}
@Override
public void unsubscribe() {
boolean unsubscribe = false;
synchronized (this) {
if (!isUnsubscribed) {
unsubscribe = true;
isUnsubscribed = true;
isTerminated = true;
}
}
if (unsubscribe) {
postTerminate();
}
}
@Override
public boolean isUnsubscribed() {
return isUnsubscribed;
}
@Override
public void request(final long n) {
if (n < 1) {
throw new IllegalArgumentException("Number requested cannot be negative: " + n);
}
boolean requestData = false;
synchronized (this) {
if (requested + n < 1) {
requested = Long.MAX_VALUE;
} else {
requested += n;
}
if (!requestedData) {
requestedData = true;
requestData = true;
}
}
if (requestData) {
requestInitialData();
} else {
processResultsQueue();
}
}
abstract void requestInitialData();
void requestMoreData() {
}
void postTerminate() {
}
abstract boolean checkCompleted();
boolean isTerminated() {
return isTerminated;
}
long getRequested() {
return requested;
}
void addToQueue(final TResult result) {
if (result != null) {
resultsQueue.add(result);
}
processResultsQueue();
}
void addToQueue(final List results) {
if (results != null) {
resultsQueue.addAll(results);
}
processResultsQueue();
}
void onError(final Throwable t) {
if (terminalAction()) {
postTerminate();
observer.onError(t);
}
}
void onNext(final TResult next) {
boolean isTerminated = false;
synchronized (this) {
isTerminated = this.isTerminated;
}
if (!isTerminated) {
try {
observer.onNext(next);
} catch (Throwable t) {
onError(t);
}
}
}
void onComplete() {
if (terminalAction()) {
postTerminate();
observer.onComplete();
}
}
private void processResultsQueue() {
boolean mustProcess = false;
synchronized (this) {
if (!isProcessing && !isTerminated) {
isProcessing = true;
mustProcess = true;
}
}
if (mustProcess) {
boolean requestMore = false;
long processedCount = 0;
boolean completed = false;
while (true) {
long localWanted = 0;
synchronized (this) {
requested -= processedCount;
if (resultsQueue.isEmpty()) {
completed = checkCompleted();
requestMore = requested > 0;
isProcessing = false;
break;
} else if (requested == 0) {
isProcessing = false;
break;
}
localWanted = requested;
}
processedCount = 0;
while (localWanted > 0) {
TResult item = resultsQueue.poll();
if (item == null) {
break;
} else {
onNext(item);
localWanted -= 1;
processedCount += 1;
}
}
}
if (completed) {
onComplete();
} else if (requestMore) {
requestMoreData();
}
}
}
private boolean terminalAction() {
boolean isTerminal = false;
synchronized (this) {
if (!isTerminated) {
isTerminated = true;
isTerminal = true;
}
}
return isTerminal;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy