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

io.activej.rpc.client.sender.strategy.impl.RoundRobin Maven / Gradle / Ivy

/*
 * Copyright (C) 2020 ActiveJ LLC.
 *
 * 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.activej.rpc.client.sender.strategy.impl;

import io.activej.async.callback.Callback;
import io.activej.common.annotation.ExposedInternals;
import io.activej.common.builder.AbstractBuilder;
import io.activej.rpc.client.RpcClientConnectionPool;
import io.activej.rpc.client.sender.RpcSender;
import io.activej.rpc.client.sender.Utils;
import io.activej.rpc.client.sender.strategy.RpcStrategy;
import org.jetbrains.annotations.Nullable;

import java.net.InetSocketAddress;
import java.util.List;
import java.util.Set;

@ExposedInternals
public final class RoundRobin implements RpcStrategy {
	public final List list;
	public int minActiveSubStrategies;

	public RoundRobin(List list, int minActiveSubStrategies) {
		this.list = list;
		this.minActiveSubStrategies = minActiveSubStrategies;
	}

	public static RoundRobin create(RpcStrategy... strategies) {
		return builder(strategies).build();
	}

	public static RoundRobin create(List strategies) {
		return builder(strategies).build();
	}

	public static Builder builder(RpcStrategy... strategies) {
		return builder(List.of(strategies));
	}

	public static Builder builder(List strategies) {
		return new RoundRobin(strategies, 0).new Builder();
	}

	public final class Builder extends AbstractBuilder {
		private Builder() {}

		public Builder withMinActiveSubStrategies(int minActiveSubStrategies) {
			checkNotBuilt(this);
			RoundRobin.this.minActiveSubStrategies = minActiveSubStrategies;
			return this;
		}

		@Override
		protected RoundRobin doBuild() {
			return RoundRobin.this;
		}
	}

	@Override
	public Set getAddresses() {
		return Utils.getAddresses(list);
	}

	@Override
	public @Nullable RpcSender createSender(RpcClientConnectionPool pool) {
		List subSenders = Utils.listOfSenders(list, pool);
		if (subSenders.size() < minActiveSubStrategies)
			return null;
		if (subSenders.isEmpty())
			return null;
		if (subSenders.size() == 1)
			return subSenders.get(0);
		return new Sender(subSenders);
	}

	public static final class Sender implements RpcSender {
		private int nextSender;
		private final RpcSender[] subSenders;

		Sender(List senders) {
			assert !senders.isEmpty();
			this.subSenders = senders.toArray(new RpcSender[0]);
			this.nextSender = 0;
		}

		@Override
		public  void sendRequest(I request, int timeout, Callback cb) {
			RpcSender sender = subSenders[nextSender];
			nextSender = (nextSender + 1) % subSenders.length;
			sender.sendRequest(request, timeout, cb);
		}

	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy