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

io.smallrye.stork.api.Service Maven / Gradle / Ivy

Go to download

Main Stork API classes. You are likely to need `smallrye-stork-core` and not this module.

The newest version!
package io.smallrye.stork.api;

import java.util.Collection;
import java.util.List;
import java.util.concurrent.Semaphore;

import io.smallrye.mutiny.Uni;
import io.smallrye.stork.api.observability.ObservationCollector;
import io.smallrye.stork.api.observability.StorkObservation;

/**
 * Represents a Service.
 * 

* This container gives you access to the service instances and to the selected service instance. */ public class Service { private final Semaphore instanceSelectionLock; private final LoadBalancer loadBalancer; private final ServiceDiscovery serviceDiscovery; private final ServiceRegistrar serviceRegistrar; private final String serviceName; private final String serviceDiscoveryType; private final String serviceSelectionType; private final ObservationCollector observations; /** * Creates a new Service. * * @param serviceName the name, must not be {@code null}, must not be blank * @param serviceDiscoveryType the type of the service discovery (for observability purpose) * @param serviceSelectionType the type of the service selection (for observability purpose) * @param collector the observation collector, must not be {@code null} * @param loadBalancer the load balancer, can be {@code null} * @param serviceDiscovery the service discovery, must not be {@code null} * @param serviceRegistrar the service registrar, can be {@code null} * @param requiresStrictRecording whether strict recording must be enabled */ public Service(String serviceName, String serviceSelectionType, String serviceDiscoveryType, ObservationCollector collector, LoadBalancer loadBalancer, ServiceDiscovery serviceDiscovery, ServiceRegistrar serviceRegistrar, boolean requiresStrictRecording) { this.loadBalancer = loadBalancer; this.serviceDiscovery = serviceDiscovery; this.serviceRegistrar = serviceRegistrar; this.serviceDiscoveryType = serviceDiscoveryType; this.serviceSelectionType = serviceSelectionType; this.observations = collector; this.serviceName = serviceName; this.instanceSelectionLock = requiresStrictRecording ? new Semaphore(1) : null; } private Service(Builder builder) { this.serviceName = builder.serviceName; this.serviceSelectionType = builder.serviceSelectionType; this.serviceDiscoveryType = builder.serviceDiscoveryType; this.observations = builder.observations; this.loadBalancer = builder.loadBalancer; this.serviceDiscovery = builder.serviceDiscovery; this.serviceRegistrar = builder.serviceRegistrar; this.instanceSelectionLock = builder.requiresStrictRecording ? new Semaphore(1) : null; } /** * Selects a service instance. *

* The selection looks for the service instances and select the one to use using the load balancer. *

* Note: this method doesn't record a start of an operation using this load balancer and does not * synchronize load balancer invocations even if the load balancer is not thread safe * * @return a Uni with a ServiceInstance, or with {@link NoServiceInstanceFoundException} if the load balancer failed to find * a service instance capable of handling a call */ public Uni selectInstance() { StorkObservation observationPoints = observations.create(serviceName, serviceDiscoveryType, serviceSelectionType); return serviceDiscovery.getServiceInstances() .onItemOrFailure().invoke((list, failure) -> { if (failure != null) { observationPoints.onServiceDiscoveryFailure(failure); } else { observationPoints.onServiceDiscoverySuccess(list); } }) .map(this::selectInstance) .onItemOrFailure().invoke((selected, failure) -> { if (failure != null) { observationPoints.onServiceSelectionFailure(failure); } else { observationPoints.onServiceSelectionSuccess(selected.getId()); } }); } /** * Using the underlying load balancer, select a service instance from the collection of service instances. *

* Note: this method doesn't record a start of an operation using this load balancer and does not * synchronize load balancer invocations even if the load balancer is not thread safe * * @param instances collection of instances * @return a ServiceInstance, or with {@link NoServiceInstanceFoundException} if the load balancer failed to find * a service instance capable of handling a call * @throws NoServiceInstanceFoundException if all the instances are nto */ public ServiceInstance selectInstance(Collection instances) { return loadBalancer.selectServiceInstance(instances); } /** * Selects a service instance and records a start of an operation using the instance *

* The selection looks for the service instances and select the one to use using the load balancer. * * @param measureTime whether the operation for which the operation is chosen records time (will call * {@link ServiceInstance#recordReply()}) * @return a Uni with a ServiceInstance, or with {@link NoServiceInstanceFoundException} if the load balancer failed to find * a service instance capable of handling a call * @see LoadBalancer#requiresStrictRecording() */ public Uni selectInstanceAndRecordStart(boolean measureTime) { StorkObservation observationPoints = observations.create(serviceName, serviceDiscoveryType, serviceSelectionType); return serviceDiscovery.getServiceInstances().onItemOrFailure().invoke((list, failure) -> { if (failure != null) { observationPoints.onServiceDiscoveryFailure(failure); } else { observationPoints.onServiceDiscoverySuccess(list); } }) .map(list -> selectInstanceAndRecordStart(list, measureTime)) .onItemOrFailure().invoke((selected, failure) -> { if (failure != null) { observationPoints.onServiceSelectionFailure(failure); } else { observationPoints.onServiceSelectionSuccess(selected.getId()); } }); } /** * Select a ServiceInstance for this service from a collection and record a start of an operation using the instance. *

* Access to the underlying LoadBalancer method is serialized. * * @param instances instances to choose from * @param measureTime whether the operation for which the operation is chosen records time (will call * {@link ServiceInstance#recordReply()}) * @return the selected instance * @see #selectInstanceAndRecordStart(boolean) * @see LoadBalancer#requiresStrictRecording() */ public ServiceInstance selectInstanceAndRecordStart(Collection instances, boolean measureTime) { if (instanceSelectionLock == null) { ServiceInstance result = loadBalancer.selectServiceInstance(instances); if (result.gatherStatistics()) { result.recordStart(measureTime); } return result; } else { try { instanceSelectionLock.acquire(); try { ServiceInstance result = loadBalancer.selectServiceInstance(instances); if (result.gatherStatistics()) { result.recordStart(measureTime); } return result; } finally { instanceSelectionLock.release(); } } catch (InterruptedException e) { Thread.currentThread().interrupt(); throw new RuntimeException("Failed to lock for ServiceInstance selection", e); } } } /** * Provide a collection of available {@link ServiceInstance}s * * @return a Uni producing the list of {@link ServiceInstance ServiceInstances}. */ public Uni> getInstances() { return serviceDiscovery.getServiceInstances(); } /** * Get the underlying load balancer instance. * * @return load balancer */ public LoadBalancer getLoadBalancer() { return loadBalancer; } /** * Get the underlying service discovery * * @return service discovery */ public ServiceDiscovery getServiceDiscovery() { return serviceDiscovery; } /** * Get the underlying service registrar * * @return service registrar */ public ServiceRegistrar getServiceRegistrar() { return serviceRegistrar; } public ObservationCollector getObservations() { return observations; } /** * @return the service name. */ public String getServiceName() { return serviceName; } public static class Builder { private String serviceName; private final ObservationCollector observations; private ServiceDiscovery serviceDiscovery; private String serviceSelectionType = "round-robin"; private String serviceDiscoveryType; private LoadBalancer loadBalancer; private ServiceRegistrar serviceRegistrar; private boolean requiresStrictRecording = false; public Builder serviceName(String serviceName) { this.serviceName = serviceName; return this; } public Builder(ObservationCollector observations) { this.observations = observations; } public Builder serviceDiscovery(ServiceDiscovery serviceDiscovery) { this.serviceDiscovery = serviceDiscovery; return this; } public Builder serviceSelectionType(String serviceSelectionType) { this.serviceSelectionType = serviceSelectionType; return this; } public Builder serviceDiscoveryType(String serviceDiscoveryType) { this.serviceDiscoveryType = serviceDiscoveryType; return this; } public Builder loadBalancer(LoadBalancer loadBalancer) { this.loadBalancer = loadBalancer; return this; } public Builder serviceRegistrar(ServiceRegistrar serviceRegistrar) { this.serviceRegistrar = serviceRegistrar; return this; } public Builder requiresStrictRecording(boolean requiresStrictRecording) { this.requiresStrictRecording = requiresStrictRecording; return this; } public Service build() { return new Service(this); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy