package io.micronaut.http.netty.channel;
import edu.umd.cs.findbugs.annotations.NonNull;
import io.micronaut.context.BeanLocator;
import io.micronaut.context.annotation.*;
import io.micronaut.context.exceptions.ConfigurationException;
import io.micronaut.core.annotation.Internal;
import io.micronaut.core.util.ArgumentUtils;
import io.micronaut.inject.qualifiers.Qualifiers;
import io.netty.channel.EventLoopGroup;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.util.concurrent.DefaultThreadFactory;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.annotation.PreDestroy;
import javax.inject.Named;
import javax.inject.Singleton;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Executor;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
@Factory
@Internal
@BootstrapContextCompatible
public class DefaultEventLoopGroupRegistry implements EventLoopGroupRegistry {
private static final Logger LOG = LoggerFactory.getLogger(DefaultEventLoopGroupRegistry.class);
private final EventLoopGroupFactory eventLoopGroupFactory;
private final BeanLocator beanLocator;
private final Map<EventLoopGroup, EventLoopGroupConfiguration> eventLoopGroups = new ConcurrentHashMap<>();
public DefaultEventLoopGroupRegistry(EventLoopGroupFactory eventLoopGroupFactory, BeanLocator beanLocator) {
this.eventLoopGroupFactory = eventLoopGroupFactory;
this.beanLocator = beanLocator;
}
@PreDestroy
void shutdown() {
eventLoopGroups.forEach((eventLoopGroup, configuration) -> {
try {
long quietPeriod = configuration.getShutdownQuietPeriod().toMillis();
long timeout = configuration.getShutdownTimeout().toMillis();
eventLoopGroup.shutdownGracefully(quietPeriod, timeout, TimeUnit.MILLISECONDS);
} catch (Throwable t) {
if (LOG.isWarnEnabled()) {
LOG.warn("Error shutting down EventLoopGroup: {}", t.getMessage(), t);
}
}
});
eventLoopGroups.clear();
}
@EachBean(EventLoopGroupConfiguration.class)
@Bean
@BootstrapContextCompatible
protected EventLoopGroup eventLoopGroup(EventLoopGroupConfiguration configuration) {
final String executor = configuration.getExecutorName().orElse(null);
EventLoopGroup eventLoopGroup;
if (executor != null) {
eventLoopGroup = beanLocator.findBean(Executor.class, Qualifiers.byName(executor))
.map(executorService -> eventLoopGroupFactory.createEventLoopGroup(
configuration.getNumThreads(),
executorService,
configuration.getIoRatio().orElse(null)
)).orElseThrow(() -> new ConfigurationException("No executor service configured for name: " + executor));
} else {
ThreadFactory threadFactory = beanLocator.findBean(ThreadFactory.class, Qualifiers.byName(configuration.getName()))
.orElseGet(() -> new DefaultThreadFactory(configuration.getName() + "-" + DefaultThreadFactory.toPoolName(NioEventLoopGroup.class)));
eventLoopGroup = eventLoopGroupFactory.createEventLoopGroup(configuration, threadFactory);
}
eventLoopGroups.put(eventLoopGroup, configuration);
return eventLoopGroup;
}
@Singleton
@Requires(missingProperty = EventLoopGroupConfiguration.DEFAULT_LOOP)
@Primary
@Bean
@BootstrapContextCompatible
protected EventLoopGroup defaultEventLoopGroup(@Named(NettyThreadFactory.NAME) ThreadFactory threadFactory) {
EventLoopGroupConfiguration configuration = new DefaultEventLoopGroupConfiguration();
EventLoopGroup eventLoopGroup = eventLoopGroupFactory.createEventLoopGroup(configuration, threadFactory);
eventLoopGroups.put(eventLoopGroup, configuration);
return eventLoopGroup;
}
@NonNull
@Override
public EventLoopGroup getDefaultEventLoopGroup() {
return beanLocator.getBean(EventLoopGroup.class);
}
@Override
public Optional<EventLoopGroup> getEventLoopGroup(@NonNull String name) {
ArgumentUtils.requireNonNull("name", name);
if (EventLoopGroupConfiguration.DEFAULT.equals(name)) {
return beanLocator.findBean(EventLoopGroup.class);
} else {
return beanLocator.findBean(EventLoopGroup.class, Qualifiers.byName(name));
}
}
@Override
public Optional<EventLoopGroupConfiguration> getEventLoopGroupConfiguration(@NonNull String name) {
ArgumentUtils.requireNonNull("name", name);
return beanLocator.findBean(EventLoopGroupConfiguration.class, Qualifiers.byName(name));
}
}