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

com.alipay.sofa.runtime.impl.ComponentManagerImpl 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 com.alipay.sofa.runtime.impl;

import com.alipay.sofa.boot.log.ErrorCode;
import com.alipay.sofa.boot.log.SofaBootLoggerFactory;
import com.alipay.sofa.runtime.api.ServiceRuntimeException;
import com.alipay.sofa.runtime.api.component.ComponentName;
import com.alipay.sofa.runtime.context.SpringContextComponent;
import com.alipay.sofa.runtime.model.ComponentStatus;
import com.alipay.sofa.runtime.model.ComponentType;
import com.alipay.sofa.runtime.spi.client.ClientFactoryInternal;
import com.alipay.sofa.runtime.spi.component.ComponentInfo;
import com.alipay.sofa.runtime.spi.component.ComponentManager;
import com.alipay.sofa.runtime.spi.component.SofaRuntimeContext;
import org.slf4j.Logger;
import org.springframework.context.ApplicationContext;

import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

/**
 * Default implementation of {@link ComponentManager}.
 *
 * @author xuanbei 18/3/9
 */
@SuppressWarnings("unchecked")
public class ComponentManagerImpl implements ComponentManager {

    private static final Logger                                               LOGGER = SofaBootLoggerFactory
                                                                                         .getLogger(ComponentManager.class);
    /** container for all components */
    protected ConcurrentMap                     registry;
    /** container for resolved components */
    protected ConcurrentMap> resolvedRegistry;
    /** client factory */
    private ClientFactoryInternal                                             clientFactoryInternal;

    private SofaRuntimeContext                                                sofaRuntimeContext;

    private final ClassLoader                                                 appClassLoader;

    public ComponentManagerImpl(ClientFactoryInternal clientFactoryInternal,
                                ClassLoader appClassLoader) {
        this.registry = new ConcurrentHashMap<>(16);
        this.resolvedRegistry = new ConcurrentHashMap<>(16);
        this.clientFactoryInternal = clientFactoryInternal;
        this.appClassLoader = appClassLoader;
    }

    public Collection getComponentInfos() {
        return new ArrayList<>(registry.values());
    }

    public Collection getPendingComponentInfos() {
        List names = new ArrayList<>();
        for (ComponentInfo ri : registry.values()) {
            if (ri.getState() == ComponentStatus.REGISTERED) {
                names.add(ri.getName());
            }
        }
        return names;
    }

    @Override
    public ComponentInfo getComponentInfo(ComponentName name) {
        return registry.get(name);
    }

    @Override
    public boolean isRegistered(ComponentName name) {
        return registry.containsKey(name);
    }

    @Override
    public Collection getComponents() {
        return registry.values();
    }

    @Override
    public int size() {
        return registry.size();
    }

    @Override
    public void shutdown() {
        if (sofaRuntimeContext.getProperties().isSkipAllComponentShutdown()) {
            return;
        }
        List elems = new ArrayList<>(registry.values());
        // shutdown spring contexts first
        List springContextComponents = elems.stream()
                .filter(componentInfo -> componentInfo instanceof SpringContextComponent).toList();

        for (ComponentInfo ri : springContextComponents) {
            try {
                unregister(ri);
            } catch (Throwable t) {
                LOGGER.error(ErrorCode.convert("01-03001", ri.getName()), t);
            }
        }

        if (!springContextComponents.isEmpty()) {
            elems.removeAll(springContextComponents);
        }

        if (sofaRuntimeContext.getProperties().isSkipCommonComponentShutdown()) {
            return;
        }
        // shutdown remaining components
        for (ComponentInfo ri : elems) {
            try {
                unregister(ri);
            } catch (Throwable t) {
                LOGGER.error(ErrorCode.convert("01-03001", ri.getName()), t);
            }
        }

        try {
            if (registry != null) {
                registry.clear();
            }
            if (resolvedRegistry != null) {
                resolvedRegistry.clear();
            }
            clientFactoryInternal = null;
        } catch (Throwable t) {
            LOGGER.error(ErrorCode.convert("01-03000"), t);
        }
    }

    @Override
    public Collection getComponentTypes() {
        return resolvedRegistry.keySet();
    }

    @Override
    public void register(ComponentInfo componentInfo) {
        doRegister(componentInfo);
    }

    @Override
    public ComponentInfo registerAndGet(ComponentInfo componentInfo) {
        return doRegister(componentInfo);
    }

    @Override
    public void registerComponentClient(Class clientType, Object client) {
        clientFactoryInternal.registerClient(clientType, client);
    }

    public void setSofaRuntimeContext(SofaRuntimeContext sofaRuntimeContext) {
        this.sofaRuntimeContext = sofaRuntimeContext;
    }

    private ComponentInfo doRegister(ComponentInfo ci) {
        ComponentName name = ci.getName();
        if (isRegistered(name)) {
            LOGGER.warn("Component was already registered: {}", name);
            if (ci.canBeDuplicate()) {
                return getComponentInfo(name);
            }
            throw new ServiceRuntimeException(ErrorCode.convert("01-03002", name));
        }

        try {
            ci.register();
        } catch (Throwable t) {
            LOGGER.error(ErrorCode.convert("01-03003", ci.getName()), t);
            return null;
        }

        LOGGER.info("Registering component: {}", ci.getName());

        try {
            ComponentInfo old = registry.putIfAbsent(ci.getName(), ci);
            if (old != null) {
                LOGGER.warn("Component was already registered: {}", name);
                if (ci.canBeDuplicate()) {
                    return old;
                }
                throw new ServiceRuntimeException(ErrorCode.convert("01-03002", name));

            }
            if (ci.resolve()) {
                typeRegistry(ci);
                ci.activate();
            }
        } catch (Throwable t) {
            ci.exception(new Exception(t));
            LOGGER.error(ErrorCode.convert("01-03004", ci.getName()), t);
        }

        return ci;
    }

    @Override
    public void unregister(ComponentInfo componentInfo) throws ServiceRuntimeException {
        ComponentName componentName = componentInfo.getName();
        registry.remove(componentName);

        if (componentName != null) {
            ComponentType componentType = componentName.getType();

            Map typesRi = resolvedRegistry.get(componentType);
            if (typesRi != null) {
                typesRi.remove(componentName);
            }
        }

        componentInfo.unregister();
    }

    @Override
    public Collection getComponentInfosByType(ComponentType type) {
        List componentInfos = new ArrayList<>();

        for (ComponentInfo componentInfo : registry.values()) {
            if (type.equals(componentInfo.getType())) {
                componentInfos.add(componentInfo);
            }
        }

        return componentInfos;
    }

    @Override
    public void resolvePendingResolveComponent(ComponentName componentName) {
        ComponentInfo componentInfo = registry.get(componentName);

        if (componentInfo.isResolved()) {
            return;
        }

        if (componentInfo.resolve()) {
            typeRegistry(componentInfo);
            try {
                componentInfo.activate();
            } catch (Throwable t) {
                componentInfo.exception(new Exception(t));
                LOGGER.error(ErrorCode.convert("01-03005", componentInfo.getName()), t);
            }
        }
    }

    @Override
    public Collection getComponentInfosByApplicationContext(ApplicationContext application) {
        List componentInfos = new ArrayList<>();

        for (ComponentInfo componentInfo : registry.values()) {
            if (Objects.equals(application, componentInfo.getApplicationContext())) {
                componentInfos.add(componentInfo);
            }
        }

        return componentInfos;
    }

    private void typeRegistry(ComponentInfo componentInfo) {
        ComponentName name = componentInfo.getName();
        if (name != null) {
            ComponentType type = name.getType();
            Map typesRi = resolvedRegistry.get(type);

            if (typesRi == null) {
                resolvedRegistry.putIfAbsent(type, new HashMap());
                typesRi = resolvedRegistry.get(type);
            }

            typesRi.put(name, componentInfo);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy