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

org.apache.camel.spring.cloud.CamelSpringCloudServiceRegistry Maven / Gradle / Ivy

/**
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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 org.apache.camel.spring.cloud;

import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Objects;

import org.apache.camel.cloud.ServiceDefinition;
import org.apache.camel.impl.cloud.AbstractServiceRegistry;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.cloud.client.serviceregistry.Registration;
import org.springframework.cloud.client.serviceregistry.ServiceRegistry;
import org.springframework.core.convert.ConversionService;

public class CamelSpringCloudServiceRegistry extends AbstractServiceRegistry {
    private static final Logger LOGGER = LoggerFactory.getLogger(CamelSpringCloudServiceRegistry.class);

    private final List definitions;
    private final List conversionServices;
    private final ServiceRegistry serviceRegistry;
    private final Class registrationType;

    public CamelSpringCloudServiceRegistry(Collection conversionServices, ServiceRegistry serviceRegistry) {
        this.definitions = new ArrayList<>();
        this.conversionServices = new ArrayList<>(conversionServices);
        this.serviceRegistry = serviceRegistry;
        this.registrationType = determineRegistrationType("register");
    }

    @Override
    public void register(ServiceDefinition definition) {
        synchronized (this) {
            // keep track of registered definition to remove them upon registry
            // shutdown
            if (definitions.stream().noneMatch(d -> matchById(d, definition))) {
                LOGGER.debug("Register service with definition: {} with registrations: {}", definition, registrationType);

                // compute registration from definition
                Registration result = convertServiceDefinition(definition);

                serviceRegistry.register(result);

                definitions.add(definition);
            }
        }
    }

    @Override
    public void deregister(ServiceDefinition definition) {
        synchronized (this) {
            if (definitions.stream().noneMatch(d -> matchById(d, definition))) {
                LOGGER.debug("Deregister service with definition: {} with registrations: {}", definition, registrationType);
                
                // compute registration from definition
                Registration result = convertServiceDefinition(definition);

                serviceRegistry.deregister(result);
            }

            // remove any instance with the same id
            definitions.removeIf(d -> matchById(d, definition));
        }
    }

    @Override
    protected void doStart() throws Exception {
    }

    @Override
    protected void doStop() throws Exception {
        synchronized (this) {
            new ArrayList<>(definitions).forEach(this::deregister);
        }
    }

    public ServiceRegistry getNativeServiceRegistry() {
        return this.serviceRegistry;
    }

    public > T getNativeServiceRegistry(Class type) {
        return type.cast(this.serviceRegistry);
    }

    /**
     * Determine the native registration type. This is needed because the registry
     * specific implementation provided by spring-cloud-xyz does not handle generic
     * Registration object but needs a Registration specific to the underlying
     * technology used.
     *
     * @return the registration type
     */
    private Class determineRegistrationType(String methodName) {
        Class type = null;
        Method[] methods = serviceRegistry.getClass().getDeclaredMethods();

        for (Method method: methods) {
            if (!methodName.equals(method.getName())) {
                continue;
            }

            if (method.getParameterCount() != 1) {
                continue;
            }

            Class parameterType = method.getParameterTypes()[0];
            if (Registration.class.isAssignableFrom(parameterType)) {
                if (type == null) {
                    type = (Class)parameterType;
                } else {
                    if (type.isAssignableFrom(parameterType)) {
                        type = (Class)parameterType;
                    }
                }
            }
        }

        return type != null ? type : Registration.class;
    }

    private boolean matchById(ServiceDefinition definition, ServiceDefinition reference) {
        if (definition.getId() == null || reference.getId() == null) {
            return false;
        }

        return Objects.equals(definition.getId(), reference.getId());
    }

    private Registration convertServiceDefinition(ServiceDefinition definition) {
        for (int i = 0; i < conversionServices.size(); i++) {
            ConversionService cs = conversionServices.get(i);

            if (cs.canConvert(ServiceDefinition.class, registrationType)) {
                return cs.convert(definition, registrationType);
            }
        }

        throw new IllegalStateException("Unable to convert service definition to native registration of type:" + registrationType);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy