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

org.apache.servicecomb.foundation.common.utils.SPIServiceUtils 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.servicecomb.foundation.common.utils;

import java.lang.reflect.Method;
import java.util.AbstractMap.SimpleEntry;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Comparator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.ServiceLoader;
import java.util.concurrent.ConcurrentHashMap;
import java.util.function.Function;
import java.util.stream.Collectors;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.util.ReflectionUtils;

/**
 * SPI Service utils
 *
 *
 */
public final class SPIServiceUtils {
  private static final Logger LOGGER = LoggerFactory.getLogger(SPIServiceUtils.class);

  // load one service, maybe trigger load another service
  // computeIfAbsent can not support this feature
  // so use double check
  private static final Object LOCK = new Object();

  private static final Map, List> cache = new ConcurrentHashMap<>();

  private SPIServiceUtils() {

  }

  /**
   * no cache, return new instances every time.
   */
  public static  List loadSortedService(Class serviceType) {
    List> serviceEntries = new ArrayList<>();
    ServiceLoader serviceLoader = ServiceLoader.load(serviceType);
    serviceLoader.forEach(service -> {
      int serviceOrder = 0;
      Method getOrder = ReflectionUtils.findMethod(service.getClass(), "getOrder");
      if (getOrder != null) {
        serviceOrder = (int) ReflectionUtils.invokeMethod(getOrder, service);
      }

      Entry entry = new SimpleEntry<>(serviceOrder, service);
      serviceEntries.add(entry);
    });

    List services = serviceEntries.stream()
        .sorted(Comparator.comparingInt(Entry::getKey))
        .map(Entry::getValue)
        .collect(Collectors.toList());

    StringBuilder info = new StringBuilder();
    for (int idx = 0; idx < services.size(); idx++) {
      T service = services.get(idx);
      info.append("{").append(idx).append(",").append(service.getClass().getSimpleName()).append("}");
    }
    LOGGER.info("Found SPI service {}, services={}.", serviceType.getSimpleName(), info);

    return services;
  }

  @SuppressWarnings("unchecked")
  public static  List getOrLoadSortedService(Class serviceType) {
    List services = cache.get(serviceType);
    if (services == null) {
      synchronized (LOCK) {
        services = cache.get(serviceType);
        if (services == null) {
          services = (List) loadSortedService(serviceType);
          cache.put(serviceType, services);
        }
      }
    }

    return (List) services;
  }

  /**
   * get target service.if target services are array,only random access to a service.
   */
  public static  T getTargetService(Class serviceType) {
    List services = getOrLoadSortedService(serviceType);
    if (services.isEmpty()) {
      LOGGER.info("Can not find SPI service for {}", serviceType.getName());
      return null;
    }

    return services.get(0);
  }

  public static  List getAllService(Class serviceType) {
    return getOrLoadSortedService(serviceType);
  }

  public static  List getSortedService(Class serviceType) {
    return getOrLoadSortedService(serviceType);
  }

  public static  T getPriorityHighestService(Class serviceType) {
    List services = getOrLoadSortedService(serviceType);
    if (services.isEmpty()) {
      LOGGER.info("Can not find SPI service for {}", serviceType.getName());
      return null;
    }

    return services.get(0);
  }

  public static  Collection getPriorityHighestServices(Function keyFunc, Class serviceType) {
    List services = getOrLoadSortedService(serviceType);
    if (services.isEmpty()) {
      LOGGER.info("Can not find SPI service for {}", serviceType.getName());
      return null;
    }

    Map map = new HashMap<>();
    services.forEach(instance -> map.putIfAbsent(keyFunc.apply(instance), instance));
    return map.values();
  }

  @SuppressWarnings("unchecked")
  public static  IMPL getTargetService(Class serviceType, Class implType) {
    List services = getOrLoadSortedService(serviceType);
    return (IMPL) services
        .stream()
        .filter(service -> service.getClass().equals(implType))
        .findFirst()
        .orElse(null);
  }
}