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

io.servicetalk.client.api.internal.AbstractRequestConcurrencyController Maven / Gradle / Ivy

/*
 * Copyright © 2018 Apple Inc. and the ServiceTalk project 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 io.servicetalk.client.api.internal;

import io.servicetalk.client.api.ConsumableEvent;
import io.servicetalk.concurrent.Cancellable;
import io.servicetalk.concurrent.CompletableSource;
import io.servicetalk.concurrent.api.Completable;
import io.servicetalk.concurrent.api.Publisher;
import io.servicetalk.concurrent.internal.LatestValueSubscriber;

import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;

import static io.servicetalk.concurrent.api.SourceAdapters.toSource;
import static java.util.concurrent.atomic.AtomicIntegerFieldUpdater.newUpdater;

@Deprecated // FIXME: 0.43 - remove deprecated class
abstract class AbstractRequestConcurrencyController implements RequestConcurrencyController {
    private static final AtomicIntegerFieldUpdater
            pendingRequestsUpdater = newUpdater(AbstractRequestConcurrencyController.class,
            "pendingRequests");

    private static final int STATE_QUIT = -1;

    @SuppressWarnings("unused")
    private volatile int pendingRequests;
    private final LatestValueSubscriber maxConcurrencyHolder;

    AbstractRequestConcurrencyController(final Publisher> maxConcurrency,
                                         final Completable onClosing) {
        maxConcurrencyHolder = new LatestValueSubscriber<>();
        // Subscribe to onClosing() before maxConcurrency, this order increases the chances of capturing the STATE_QUIT
        // before observing 0 from maxConcurrency which could lead to more ambiguous max concurrency error messages for
        // the users on connection tear-down.
        toSource(onClosing).subscribe(new CompletableSource.Subscriber() {
            @Override
            public void onSubscribe(Cancellable cancellable) {
                // No op
            }

            @Override
            public void onComplete() {
                pendingRequests = STATE_QUIT;
            }

            @Override
            public void onError(Throwable ignored) {
                pendingRequests = STATE_QUIT;
            }
        });
        toSource(maxConcurrency
                .afterOnNext(ConsumableEvent::eventConsumed)
                .map(ConsumableEvent::event)
        ).subscribe(maxConcurrencyHolder);
    }

    @Override
    public final void requestFinished() {
        pendingRequestsUpdater.decrementAndGet(this);
    }

    final int lastSeenMaxValue(int defaultValue) {
        return maxConcurrencyHolder.lastSeenValue(defaultValue);
    }

    final int pendingRequests() {
        return pendingRequests;
    }

    final boolean casPendingRequests(int oldValue, int newValue) {
        return pendingRequestsUpdater.compareAndSet(this, oldValue, newValue);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy