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

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

There is a newer version: 4.8.0-rc1
Show newest version
/*
 * (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.math.IntMath;
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.core.BalancedScoreTracker.ChannelScoreInfo;
import com.palantir.dialogue.core.BalancedScoreTracker.ScoreSnapshot;
import com.palantir.dialogue.futures.DialogueFutures;
import com.palantir.logsafe.Preconditions;
import com.palantir.logsafe.SafeArg;
import com.palantir.logsafe.UnsafeArg;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
import com.palantir.tritium.metrics.registry.TaggedMetricRegistry;
import java.util.Optional;
import java.util.Random;
import java.util.stream.IntStream;

/**
 * Chooses nodes to achieve the best possible client-side load balancing, by computing a 'score' for each channel and
 * always routing requests to the channel with the lowest score. See {@link BalancedScoreTracker} for details of what
 * goes into a 'score'.
 *
 * This is intended to be a strict improvement over Round Robin and Random Selection which can leave fast servers
 * underutilized, as it sends the same number to both a slow and fast node. It is *not* appropriate for transactional
 * workloads (where n requests must all land on the same server) or scenarios where cache warming is very important.
 * {@link PinUntilErrorNodeSelectionStrategyChannel} remains the best choice for these.
 */
final class BalancedNodeSelectionStrategyChannel implements LimitedChannel {
    private static final SafeLogger log = SafeLoggerFactory.get(BalancedNodeSelectionStrategyChannel.class);

    private static final int INFLIGHT_COMPARISON_THRESHOLD = 5;
    // When a channel has UNHEALTHY_SCORE_MULTIPLIER times the score of a channel with INFLIGHT_COMPARISON_THRESHOLD
    // active requests, it's considered unhealthy and may not be attempted.
    private static final int UNHEALTHY_SCORE_MULTIPLIER = 2;

    private final BalancedScoreTracker tracker;
    private final ImmutableList channels;

    BalancedNodeSelectionStrategyChannel(
            ImmutableList channels,
            Random random,
            Ticker ticker,
            TaggedMetricRegistry taggedMetrics,
            String channelName) {
        Preconditions.checkState(channels.size() >= 2, "At least two channels required");
        this.tracker = new BalancedScoreTracker(channels.size(), random, ticker, taggedMetrics, channelName);
        this.channels = IntStream.range(0, channels.size())
                .mapToObj(index -> new BalancedChannel(
                        channels.get(index), tracker.channelStats().get(index)))
                .collect(ImmutableList.toImmutableList());
        log.debug("Initialized", SafeArg.of("count", channels.size()), UnsafeArg.of("channels", channels));
    }

    @Override
    public Optional> maybeExecute(
            Endpoint endpoint, Request request, LimitEnforcement limitEnforcement) {
        ScoreSnapshot[] snapshotsByScore = tracker.getSnapshotsInOrderOfIncreasingScore();

        int giveUpThreshold = Integer.MAX_VALUE;
        for (ScoreSnapshot snapshot : snapshotsByScore) {
            /*
             * If we're considering a channel that has a *drastically* higher score than the last one (i.e. we
             * think it's much worse), then we can often get better outcomes by just refusing to send a
             * request (and queueing) rather than sending something to this known-bad channel.
             *
             * This allows us to avoid sending requests to an unhealthy channel after a node has failed while
             * the concurrency limit on the healthy channel is slowly expanded to meet increased load. Otherwise
             * the assumed concurrency limit base don lower request load on the healthy channel may result in requests
             * being sent to a node that's no longer alive.
             *
             * Note that this functionality is not safe if the preferred channel had zero inflight requests (as this
             * could result in infinite queuing).
             */
            if (snapshot.getScore() > giveUpThreshold) {
                if (log.isDebugEnabled()) {
                    log.debug(
                            "Giving up and queueing because channel score ({}) for channel {} is not worth sending a "
                                    + "request to ({})",
                            SafeArg.of("score", snapshot.getScore()),
                            SafeArg.of("hostIndex", snapshot.getDelegate().channelIndex()),
                            SafeArg.of("giveUpScore", giveUpThreshold));
                }
                return Optional.empty();
            }
            if (snapshot.getInflight() > INFLIGHT_COMPARISON_THRESHOLD) {
                int newThreshold = IntMath.saturatedMultiply(snapshot.getScore(), UNHEALTHY_SCORE_MULTIPLIER);
                if (log.isDebugEnabled()) {
                    log.debug(
                            "When considering channel {}, giveUpThreshold {} -> {}",
                            SafeArg.of("hostIndex", snapshot.getDelegate().channelIndex()),
                            SafeArg.of("old", giveUpThreshold),
                            SafeArg.of("new", newThreshold));
                }
                giveUpThreshold = newThreshold;
            }

            BalancedChannel channel = channels.get(snapshot.getDelegate().channelIndex());
            Optional> maybe =
                    StickyAttachments.maybeAddStickyToken(channel, endpoint, request, limitEnforcement);
            if (maybe.isPresent()) {
                return maybe;
            }
        }

        return Optional.empty();
    }

    private static final class BalancedChannel implements LimitedChannel {
        private final LimitedChannel delegate;
        private final ChannelScoreInfo channelInfo;

        BalancedChannel(LimitedChannel delegate, ChannelScoreInfo channelInfo) {
            this.delegate = delegate;
            this.channelInfo = channelInfo;
        }

        @Override
        public Optional> maybeExecute(
                Endpoint endpoint, Request request, LimitEnforcement limitEnforcement) {
            channelInfo.startRequest();

            Optional> maybe = delegate.maybeExecute(endpoint, request, limitEnforcement);

            if (maybe.isPresent()) {
                channelInfo.observability().markRequestMade();
                DialogueFutures.addDirectCallback(maybe.get(), channelInfo);
                return maybe;
            } else {
                channelInfo.undoStartRequest();
            }
            return Optional.empty();
        }

        @Override
        public String toString() {
            return "BalancedChannel{delegate=" + delegate + ", channelInfo=" + channelInfo + '}';
        }
    }

    @VisibleForTesting
    IntStream getScoresForTesting() {
        return tracker.getScoresForTesting();
    }

    @Override
    public String toString() {
        return "BalancedNodeSelectionStrategyChannel{channels=" + channels + ", tracker=" + tracker + '}';
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy