io.micronaut.discovery.eureka.registration.EurekaAutoRegistration Maven / Gradle / Ivy
/*
* Copyright 2017-2020 original 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
*
* https://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.micronaut.discovery.eureka.registration;
import io.micronaut.context.annotation.Requires;
import io.micronaut.context.env.Environment;
import io.micronaut.core.convert.value.ConvertibleValues;
import io.micronaut.discovery.ServiceInstance;
import io.micronaut.discovery.ServiceInstanceIdGenerator;
import io.micronaut.discovery.client.registration.DiscoveryServiceAutoRegistration;
import io.micronaut.discovery.eureka.EurekaConfiguration;
import io.micronaut.discovery.eureka.client.v2.EurekaClient;
import io.micronaut.discovery.eureka.client.v2.InstanceInfo;
import io.micronaut.health.HealthStatus;
import io.micronaut.health.HeartbeatConfiguration;
import io.micronaut.http.HttpStatus;
import io.micronaut.http.client.exceptions.HttpClientResponseException;
import io.micronaut.runtime.ApplicationConfiguration;
import io.micronaut.runtime.server.EmbeddedServer;
import jakarta.inject.Singleton;
import org.reactivestreams.Publisher;
import org.reactivestreams.Subscriber;
import org.reactivestreams.Subscription;
import reactor.core.publisher.Mono;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.atomic.AtomicReference;
/**
* A {@link io.micronaut.discovery.registration.AutoRegistration} that registers with Eureka.
*
* @author Graeme Rocher
* @since 1.0
*/
@Singleton
@Requires(beans = {EurekaClient.class, EmbeddedServer.class})
@Requires(property = ApplicationConfiguration.APPLICATION_NAME)
public class EurekaAutoRegistration extends DiscoveryServiceAutoRegistration {
private static final String EUREKA_SERVICE_NAME = "Eureka";
private final Environment environment;
private final EurekaClient eurekaClient;
private final EurekaConfiguration eurekaConfiguration;
private final HeartbeatConfiguration heartbeatConfiguration;
private final ServiceInstanceIdGenerator idGenerator;
private final AtomicReference lastStatus = new AtomicReference<>();
/**
* @param environment The environment
* @param eurekaClient The Eureka client
* @param eurekaConfiguration The Eureka configuration
* @param heartbeatConfiguration The Heartbeat configuration
* @param idGenerator The id generator
*/
protected EurekaAutoRegistration(
Environment environment,
EurekaClient eurekaClient,
EurekaConfiguration eurekaConfiguration,
HeartbeatConfiguration heartbeatConfiguration,
ServiceInstanceIdGenerator idGenerator) {
super(eurekaConfiguration.getRegistration());
this.environment = environment;
this.eurekaClient = eurekaClient;
this.eurekaConfiguration = eurekaConfiguration;
this.heartbeatConfiguration = heartbeatConfiguration;
this.idGenerator = idGenerator;
}
@Override
protected void pulsate(ServiceInstance instance, HealthStatus status) {
EurekaConfiguration.EurekaRegistrationConfiguration registration = eurekaConfiguration.getRegistration();
if (heartbeatConfiguration.isEnabled() && registration != null && registered.get()) {
InstanceInfo instanceInfo = registration.getInstanceInfo();
if (status.equals(HealthStatus.UP)) {
Mono heartbeatPublisher = Mono.from(eurekaClient.heartbeat(instanceInfo.getApp(), instanceInfo.getId()));
heartbeatPublisher.subscribe(httpStatus -> {
if (LOG.isDebugEnabled()) {
LOG.debug("Successfully reported passing state to Eureka");
}
}, throwable -> {
if (throwable instanceof HttpClientResponseException hcre) {
HttpStatus httpStatus = hcre.getStatus();
if (httpStatus == HttpStatus.NOT_FOUND) {
if (LOG.isInfoEnabled()) {
LOG.info("Instance [{}] no longer registered with Eureka. Attempting re-registration.", instance.getId());
}
register(instance);
return;
}
}
String errorMessage = getErrorMessage(throwable, "Error reporting passing state to Eureka: ");
if (LOG.isErrorEnabled()) {
LOG.error(errorMessage, throwable);
}
});
}
final HealthStatus lastStatus = this.lastStatus.getAndSet(status);
if (lastStatus == null || !lastStatus.equals(status)) {
InstanceInfo.Status s = translateState(status);
eurekaClient.updateStatus(instanceInfo.getApp(), instanceInfo.getId(), s)
.subscribe(new Subscriber<>() {
@Override
public void onSubscribe(Subscription s) {
s.request(1);
}
@Override
public void onNext(HttpStatus httpStatus) {
if (LOG.isDebugEnabled()) {
LOG.debug("Successfully reported status {} to Eureka", s);
}
}
@Override
public void onError(Throwable t) {
String errorMessage = getErrorMessage(t, "Error reporting state to Eureka: ");
if (LOG.isErrorEnabled()) {
LOG.error(errorMessage, t);
}
}
@Override
public void onComplete() {
// no-op
}
});
}
}
}
/**
* Translate a {@link HealthStatus} to a Eureka {@link io.micronaut.discovery.eureka.client.v2.InstanceInfo.Status}.
*
* @param status The status
* @return The {@link io.micronaut.discovery.eureka.client.v2.InstanceInfo.Status} instance
*/
protected InstanceInfo.Status translateState(HealthStatus status) {
if (status.equals(HealthStatus.UP)) {
return InstanceInfo.Status.UP;
}
return InstanceInfo.Status.DOWN;
}
@Override
protected void deregister(ServiceInstance instance) {
EurekaConfiguration.EurekaRegistrationConfiguration registration = eurekaConfiguration.getRegistration();
if (registration != null) {
InstanceInfo instanceInfo = registration.getInstanceInfo();
Publisher deregisterPublisher = eurekaClient.deregister(instanceInfo.getApp(), instanceInfo.getId());
performDeregistration(EUREKA_SERVICE_NAME, registration, deregisterPublisher, instanceInfo.getApp());
}
}
@Override
protected void register(ServiceInstance instance) {
EurekaConfiguration.EurekaRegistrationConfiguration registration = eurekaConfiguration.getRegistration();
if (registration != null) {
InstanceInfo instanceInfo = registration.getInstanceInfo();
if (!registration.isExplicitInstanceId()) {
instanceInfo.setInstanceId(idGenerator.generateId(environment, instance));
}
ConvertibleValues instanceMetadata = instance.getMetadata();
Properties properties = instanceMetadata.asProperties();
Map targetMetadata = instanceInfo.getMetadata();
for (Map.Entry
© 2015 - 2025 Weber Informatics LLC | Privacy Policy