io.gravitee.gateway.services.sync.cache.SubscriptionsCacheService Maven / Gradle / Ivy
/**
* Copyright (C) 2015 The Gravitee team (http://gravitee.io)
*
* 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
*
* 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 io.gravitee.gateway.services.sync.cache;
import io.gravitee.common.event.Event;
import io.gravitee.common.event.EventListener;
import io.gravitee.common.event.EventManager;
import io.gravitee.common.http.MediaType;
import io.gravitee.common.service.AbstractService;
import io.gravitee.definition.model.Plan;
import io.gravitee.gateway.handlers.api.definition.Api;
import io.gravitee.gateway.reactor.Reactable;
import io.gravitee.gateway.reactor.ReactorEvent;
import io.gravitee.gateway.services.sync.cache.handler.SubscriptionsServiceHandler;
import io.gravitee.gateway.services.sync.cache.repository.SubscriptionRepositoryWrapper;
import io.gravitee.gateway.services.sync.cache.task.FullSubscriptionRefresher;
import io.gravitee.gateway.services.sync.cache.task.IncrementalSubscriptionRefresher;
import io.gravitee.gateway.services.sync.cache.task.Result;
import io.gravitee.node.api.cluster.ClusterManager;
import io.gravitee.repository.management.api.SubscriptionRepository;
import io.vertx.ext.web.Router;
import java.time.Duration;
import java.util.*;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Function;
import java.util.stream.Collectors;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;
/**
* @author David BRASSELY (david.brassely at graviteesource.com)
* @author GraviteeSource Team
*/
public class SubscriptionsCacheService extends AbstractService implements EventListener {
private static final Logger LOGGER = LoggerFactory.getLogger(SubscriptionsCacheService.class);
private static final String CACHE_NAME = "subscriptions";
@Value("${services.sync.bulk_items:100}")
private int bulkItems;
@Value("${services.sync.distributed:false}")
private boolean distributed;
private static final String PATH = "/subscriptions";
@Autowired
private EventManager eventManager;
@Autowired
private CacheManager cacheManager;
private SubscriptionRepository subscriptionRepository;
@Autowired
@Qualifier("syncExecutor")
private ThreadPoolExecutor executorService;
@Autowired
@Qualifier("managementRouter")
private Router router;
@Autowired
private ClusterManager clusterManager;
private final ThreadPoolTaskScheduler scheduler;
private ScheduledFuture> scheduledFuture;
private final Map> plansPerApi = new ConcurrentHashMap<>();
public SubscriptionsCacheService() {
scheduler = new ThreadPoolTaskScheduler();
scheduler.setThreadNamePrefix("gio.sync-subscriptions-");
// Ensure every execution is done before running next execution
scheduler.setPoolSize(1);
scheduler.initialize();
}
@Override
protected void doStart() throws Exception {
super.doStart();
LOGGER.info("Overriding subscription repository implementation with a cached subscription repository");
DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory) (
(ConfigurableApplicationContext) applicationContext.getParent()
).getBeanFactory();
this.subscriptionRepository = beanFactory.getBean(SubscriptionRepository.class);
LOGGER.debug("Current subscription repository implementation is {}", subscriptionRepository.getClass().getName());
String[] beanNames = beanFactory.getBeanNamesForType(SubscriptionRepository.class);
String oldBeanName = beanNames[0];
beanFactory.destroySingleton(oldBeanName);
LOGGER.debug("Register subscription repository implementation {}", SubscriptionRepositoryWrapper.class.getName());
beanFactory.registerSingleton(
SubscriptionRepository.class.getName(),
new SubscriptionRepositoryWrapper(subscriptionRepository, cacheManager.getCache(CACHE_NAME))
);
LOGGER.info("Associate a new HTTP handler on {}", PATH);
// Create handlers
// Set subscriptions handler
SubscriptionsServiceHandler subscriptionsServiceHandler = new SubscriptionsServiceHandler(executorService);
applicationContext.getAutowireCapableBeanFactory().autowireBean(subscriptionsServiceHandler);
router.get(PATH).produces(MediaType.APPLICATION_JSON).handler(subscriptionsServiceHandler);
}
public void startScheduler(int delay, TimeUnit unit) {
scheduledFuture = scheduler.scheduleAtFixedRate(new SubscriptionsTask(), Duration.ofMillis(unit.toMillis(delay)));
eventManager.subscribeForEvents(this, ReactorEvent.class);
}
class SubscriptionsTask extends TimerTask {
private long lastRefreshAt = -1;
@Override
public void run() {
if (clusterManager.isMasterNode() || (!clusterManager.isMasterNode() && !distributed)) {
long nextLastRefreshAt = System.currentTimeMillis();
// Merge all plans and split them into buckets
final Set plans = plansPerApi.values().stream().flatMap(Set::stream).collect(Collectors.toSet());
final AtomicInteger counter = new AtomicInteger();
final Collection> chunks = plans
.stream()
.collect(Collectors.groupingBy(it -> counter.getAndIncrement() / bulkItems))
.values();
// Run refreshers
if (!chunks.isEmpty()) {
// Prepare tasks
final List>> callables = chunks
.stream()
.map(
new Function, IncrementalSubscriptionRefresher>() {
@Override
public IncrementalSubscriptionRefresher apply(List chunks) {
IncrementalSubscriptionRefresher refresher = new IncrementalSubscriptionRefresher(
lastRefreshAt,
nextLastRefreshAt,
chunks
);
refresher.setSubscriptionRepository(subscriptionRepository);
refresher.setCache(cacheManager.getCache(CACHE_NAME));
return refresher;
}
}
)
.collect(Collectors.toList());
// And run...
try {
List>> futures = executorService.invokeAll(callables);
boolean failure = futures
.stream()
.anyMatch(
resultFuture -> {
try {
return resultFuture.get().failed();
} catch (Exception e) {
e.printStackTrace();
}
return false;
}
);
// If there is no failure, move to the next period of time
if (!failure) {
lastRefreshAt = nextLastRefreshAt;
}
} catch (InterruptedException e) {
LOGGER.error("Unexpected error while running the subscriptions refresher");
}
} else {
lastRefreshAt = nextLastRefreshAt;
}
}
}
}
@Override
protected void doStop() throws Exception {
super.doStop();
if (scheduledFuture != null) {
scheduledFuture.cancel(false);
}
if (scheduler != null) {
scheduler.shutdown();
}
}
@Override
protected String name() {
return "Subscriptions cache repository";
}
@Override
public void onEvent(Event event) {
final Api api = (Api) event.content();
switch (event.type()) {
case DEPLOY:
register(api);
break;
case UNDEPLOY:
unregister(api);
break;
case UPDATE:
unregister(api);
register(api);
break;
default:
// Nothing to do with unknown event type
break;
}
}
private void register(Api api) {
register(Collections.singletonList(api));
}
public void register(List apis) {
final Map apisById = apis.stream().collect(Collectors.toMap(io.gravitee.definition.model.Api::getId, api -> api));
// Filters plans to update api_keys only for them
final Map> plansByApi = apis
.stream()
.filter(Api::isEnabled)
.map(
api ->
new AbstractMap.SimpleEntry<>(
api.getId(),
api
.getPlans()
.stream()
.filter(
plan ->
io.gravitee.repository.management.model.Plan.PlanSecurityType.OAUTH2
.name()
.equalsIgnoreCase(plan.getSecurity()) ||
io.gravitee.repository.management.model.Plan.PlanSecurityType.JWT
.name()
.equalsIgnoreCase(plan.getSecurity())
)
.map(Plan::getId)
.collect(Collectors.toSet())
)
)
// Remove if no plan.
.filter(e -> !e.getValue().isEmpty())
.collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));
if (!plansByApi.isEmpty()) {
final Set planIds = plansByApi.values().stream().flatMap(Collection::stream).collect(Collectors.toSet());
// If the node is not a master, we assume that the full refresh has been handle by an other node
if (clusterManager.isMasterNode() || (!clusterManager.isMasterNode() && !distributed)) {
final FullSubscriptionRefresher refresher = new FullSubscriptionRefresher(planIds);
refresher.setSubscriptionRepository(subscriptionRepository);
refresher.setCache(cacheManager.getCache(CACHE_NAME));
CompletableFuture
.supplyAsync(refresher::call, executorService)
.whenComplete(
(result, throwable) -> {
if (throwable != null) {
// An error occurs, we must try to full refresh again
register(apis);
} else {
// Once we are sure that the initial full refresh is a success, we cn move the plans to an incremental refresh
if (result.succeeded()) {
// Attach the plans to the global list
plansPerApi.putAll(plansByApi);
} else {
LOGGER.error(
"An error occurs while doing a full subscriptions refresh for APIs [{}]",
apisById.keySet(),
result.cause()
);
// If not, try to fully refresh again
register(apis);
}
}
}
);
} else {
// Keep track of all the plans to ensure that, once the node is becoming a master node, we are able
// to run incremental refresh for all the plans
plansPerApi.putAll(plansByApi);
}
}
}
private void unregister(Api api) {
plansPerApi.remove(api.getId());
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy