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

com.palantir.dialogue.core.NodeSelectionStrategyChannel Maven / Gradle / Ivy

/*
 * (c) Copyright 2020 Palantir Technologies Inc. All rights reserved.
 *
 * 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.palantir.dialogue.core;

import com.github.benmanes.caffeine.cache.Ticker;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableList;
import com.google.common.util.concurrent.FutureCallback;
import com.google.common.util.concurrent.ListenableFuture;
import com.palantir.dialogue.Endpoint;
import com.palantir.dialogue.Request;
import com.palantir.dialogue.Response;
import com.palantir.dialogue.futures.DialogueFutures;
import com.palantir.logsafe.SafeArg;
import com.palantir.logsafe.exceptions.SafeIllegalStateException;
import com.palantir.logsafe.exceptions.SafeRuntimeException;
import com.palantir.tritium.metrics.registry.TaggedMetricRegistry;
import java.util.List;
import java.util.Optional;
import java.util.Random;
import java.util.concurrent.atomic.AtomicReference;
import javax.annotation.Nullable;
import org.immutables.value.Value;

final class NodeSelectionStrategyChannel implements LimitedChannel {
    private static final String NODE_SELECTION_HEADER = "Node-Selection-Strategy";

    private final FutureCallback callback = new NodeSelectionCallback();
    private final AtomicReference nodeSelectionStrategy = new AtomicReference<>();
    private final NodeSelectionStrategyChooser strategySelector;

    private final String channelName;
    private final Random random;
    private final Ticker tick;
    private final TaggedMetricRegistry metrics;
    private final DialogueNodeselectionMetrics nodeSelectionMetrics;
    private final ImmutableList channels;

    @SuppressWarnings("NullAway")
    private final LimitedChannel delegate =
            new SupplierChannel(() -> nodeSelectionStrategy.get().channel());

    @VisibleForTesting
    NodeSelectionStrategyChannel(
            NodeSelectionStrategyChooser strategySelector,
            DialogueNodeSelectionStrategy initialStrategy,
            String channelName,
            Random random,
            Ticker tick,
            TaggedMetricRegistry metrics,
            ImmutableList channels) {
        this.strategySelector = strategySelector;
        this.channelName = channelName;
        this.random = random;
        this.tick = tick;
        this.metrics = metrics;
        this.nodeSelectionMetrics = DialogueNodeselectionMetrics.of(metrics);
        this.channels = channels;
        this.nodeSelectionStrategy.set(createNodeSelectionChannel(null, initialStrategy));
    }

    static LimitedChannel create(Config cf, ImmutableList channels) {
        if (channels.isEmpty()) {
            if (cf.clientConf().uris().isEmpty()) {
                // In this case the configuration lists no URIs. The exception should not be retryable
                // because the problem is local misconfiguration.
                return new StickyChannelHandler(
                        new ZeroUriNodeSelectionChannel(endpoint -> new SafeIllegalStateException(
                                "There are no URIs configured to handle requests",
                                SafeArg.of("channel", cf.channelName()),
                                SafeArg.of("service", endpoint.serviceName()),
                                SafeArg.of("endpoint", endpoint.endpointName()))));
            } else {
                // Channels is empty, however the raw config lists URIs. In this case DNS resolution
                // yielded no hostnames.
                return new StickyChannelHandler(
                        new ZeroUriNodeSelectionChannel(endpoint -> new SafeUnknownHostException(
                                "There were no DNS results for this host",
                                SafeArg.of("channel", cf.channelName()),
                                SafeArg.of("service", endpoint.serviceName()),
                                SafeArg.of("endpoint", endpoint.endpointName()))));
            }
        }

        if (channels.size() == 1) {
            return new StickyChannelHandler(new StickyTokenHandler(channels.get(0)));
        }

        return new NodeSelectionStrategyChannel(
                NodeSelectionStrategyChannel::getFirstKnownStrategy,
                DialogueNodeSelectionStrategy.of(cf.clientConf().nodeSelectionStrategy()),
                cf.channelName(),
                cf.random(),
                cf.ticker(),
                cf.clientConf().taggedMetricRegistry(),
                channels);
    }

    @Override
    public Optional> maybeExecute(
            Endpoint endpoint, Request request, LimitEnforcement limitEnforcement) {
        Optional> maybe =
                StickyChannelHandler.maybeExecute(delegate, endpoint, request, limitEnforcement);
        if (!maybe.isPresent()) {
            return Optional.empty();
        }

        ListenableFuture wrappedFuture = DialogueFutures.addDirectCallback(maybe.get(), callback);
        return Optional.of(wrappedFuture);
    }

    private NodeSelectionChannel createNodeSelectionChannel(
            @Nullable LimitedChannel previousNodeSelectionStrategy, DialogueNodeSelectionStrategy strategy) {
        NodeSelectionChannel.Builder channelBuilder =
                NodeSelectionChannel.builder().strategy(strategy);

        switch (strategy) {
            case PIN_UNTIL_ERROR:
            case PIN_UNTIL_ERROR_WITHOUT_RESHUFFLE:
                DialoguePinuntilerrorMetrics pinuntilerrorMetrics = DialoguePinuntilerrorMetrics.of(metrics);
                // Previously pin until error, so we should preserve our previous location
                if (previousNodeSelectionStrategy instanceof PinUntilErrorNodeSelectionStrategyChannel) {
                    PinUntilErrorNodeSelectionStrategyChannel previousPinUntilError =
                            (PinUntilErrorNodeSelectionStrategyChannel) previousNodeSelectionStrategy;
                    return channelBuilder
                            .channel(PinUntilErrorNodeSelectionStrategyChannel.of(
                                    Optional.of(previousPinUntilError.getCurrentChannel()),
                                    strategy,
                                    channels,
                                    pinuntilerrorMetrics,
                                    random,
                                    tick,
                                    channelName))
                            .build();
                } else {
                    return channelBuilder
                            .channel(PinUntilErrorNodeSelectionStrategyChannel.of(
                                    Optional.empty(),
                                    strategy,
                                    channels,
                                    pinuntilerrorMetrics,
                                    random,
                                    tick,
                                    channelName))
                            .build();
                }
            case BALANCED:
                // When people ask for 'ROUND_ROBIN', they usually just want something to load balance better.
                // We used to have a naive RoundRobinChannel, then tried RandomSelection and now use this heuristic:
                return channelBuilder
                        .channel(new BalancedNodeSelectionStrategyChannel(channels, random, tick, metrics, channelName))
                        .build();
            case UNKNOWN:
        }
        throw new SafeRuntimeException("Unknown NodeSelectionStrategy", SafeArg.of("unknown", strategy));
    }

    @VisibleForTesting
    static Optional getFirstKnownStrategy(
            List strategies) {
        for (DialogueNodeSelectionStrategy strategy : strategies) {
            if (!strategy.equals(DialogueNodeSelectionStrategy.UNKNOWN)) {
                return Optional.of(strategy);
            }
        }
        return Optional.empty();
    }

    @Override
    public String toString() {
        return "NodeSelectionStrategyChannel{" + nodeSelectionStrategy + '}';
    }

    @Value.Immutable
    interface NodeSelectionChannel {
        DialogueNodeSelectionStrategy strategy();

        LimitedChannel channel();

        class Builder extends ImmutableNodeSelectionChannel.Builder {}

        static Builder builder() {
            return new Builder();
        }
    }

    private final class NodeSelectionCallback implements FutureCallback {
        @Override
        public void onSuccess(Response result) {
            result.getFirstHeader(NODE_SELECTION_HEADER).ifPresent(this::consumeStrategy);
        }

        @Override
        public void onFailure(Throwable _unused) {}

        @SuppressWarnings("NullAway")
        private void consumeStrategy(String strategy) {
            Optional maybeStrategy =
                    strategySelector.updateAndGet(DialogueNodeSelectionStrategy.fromHeader(strategy));
            if (!maybeStrategy.isPresent()) {
                return;
            }
            DialogueNodeSelectionStrategy fromServer = maybeStrategy.get();

            // Quick check to avoid expensive CAS
            if (fromServer.equals(nodeSelectionStrategy.get().strategy())) {
                return;
            }

            nodeSelectionMetrics
                    .strategy()
                    .channelName(channelName)
                    .strategy(fromServer.toString())
                    .build()
                    .mark();
            nodeSelectionStrategy.getAndUpdate(
                    prevChannel -> createNodeSelectionChannel(prevChannel.channel(), fromServer));
        }
    }

    private static final class StickyChannelHandler implements LimitedChannel {

        private final LimitedChannel delegate;

        StickyChannelHandler(LimitedChannel delegate) {
            this.delegate = delegate;
        }

        static Optional> maybeExecute(
                LimitedChannel channel, Endpoint endpoint, Request request, LimitEnforcement limitEnforcement) {
            return StickyAttachments.maybeExecuteOnSticky(channel, endpoint, request, limitEnforcement);
        }

        @Override
        public Optional> maybeExecute(
                Endpoint endpoint, Request request, LimitEnforcement limitEnforcement) {
            return maybeExecute(delegate, endpoint, request, limitEnforcement);
        }
    }

    private static final class StickyTokenHandler implements LimitedChannel {

        private final LimitedChannel delegate;

        StickyTokenHandler(LimitedChannel delegate) {
            this.delegate = delegate;
        }

        @Override
        public Optional> maybeExecute(
                Endpoint endpoint, Request request, LimitEnforcement limitEnforcement) {
            return StickyAttachments.maybeAddStickyToken(delegate, endpoint, request, limitEnforcement);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy