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

com.linecorp.armeria.spring.AbstractArmeriaAutoConfiguration Maven / Gradle / Ivy

/*
 * Copyright 2020 LINE Corporation
 *
 * LINE Corporation 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:
 *
 *   https://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.linecorp.armeria.spring;

import static com.linecorp.armeria.internal.spring.ArmeriaConfigurationUtil.configureServerWithArmeriaSettings;

import java.time.Duration;
import java.util.List;
import java.util.Optional;
import java.util.function.Consumer;

import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.context.SmartLifecycle;
import org.springframework.context.annotation.Bean;

import com.google.common.collect.ImmutableList;

import com.linecorp.armeria.common.DependencyInjector;
import com.linecorp.armeria.common.SessionProtocol;
import com.linecorp.armeria.common.annotation.Nullable;
import com.linecorp.armeria.common.metric.MeterIdPrefixFunction;
import com.linecorp.armeria.server.Server;
import com.linecorp.armeria.server.ServerBuilder;
import com.linecorp.armeria.server.ServerPort;
import com.linecorp.armeria.server.docs.DocService;
import com.linecorp.armeria.server.healthcheck.HealthCheckService;
import com.linecorp.armeria.server.healthcheck.HealthChecker;
import com.linecorp.armeria.server.metric.PrometheusExpositionService;
import com.linecorp.armeria.spring.ArmeriaSettings.Port;

import io.micrometer.core.instrument.MeterRegistry;
import io.micrometer.core.instrument.Metrics;

/**
 * Abstract class for implementing ArmeriaAutoConfiguration of boot2-autoconfigure module
 * and ArmeriaSpringBoot1AutoConfiguration of boot1-autoconfigure module.
 */
public abstract class AbstractArmeriaAutoConfiguration {

    private static final Port DEFAULT_PORT = new Port().setPort(8080)
                                                       .setProtocol(SessionProtocol.HTTP);

    private static final String GRACEFUL_SHUTDOWN = "graceful";

    /**
     * Creates a started {@link Server} bean.
     */
    @Bean
    @ConditionalOnMissingBean(Server.class)
    public Server armeriaServer(
            ArmeriaSettings armeriaSettings,
            InternalServices internalService,
            Optional meterRegistry,
            Optional> metricCollectingServiceConfigurators,
            Optional meterIdPrefixFunction,
            Optional> armeriaServerConfigurators,
            Optional>> armeriaServerBuilderConsumers,
            Optional> dependencyInjectors,
            BeanFactory beanFactory) {

        if (!armeriaServerConfigurators.isPresent() &&
            !armeriaServerBuilderConsumers.isPresent()) {
            throw new IllegalStateException(
                    "No services to register, " +
                    "use ArmeriaServerConfigurator or Consumer to configure an Armeria server.");
        }

        final ServerBuilder serverBuilder = Server.builder();

        final List ports = armeriaSettings.getPorts();
        if (ports.isEmpty()) {
            assert DEFAULT_PORT.getProtocols() != null;
            serverBuilder.port(new ServerPort(DEFAULT_PORT.getPort(), DEFAULT_PORT.getProtocols()));
        }

        configureServerWithArmeriaSettings(serverBuilder, armeriaSettings, internalService,
                                           armeriaServerConfigurators.orElse(ImmutableList.of()),
                                           armeriaServerBuilderConsumers.orElse(ImmutableList.of()),
                                           meterRegistry.orElse(Metrics.globalRegistry),
                                           meterIdPrefixFunction.orElse(
                                                   MeterIdPrefixFunction.ofDefault("armeria.server")),
                                           metricCollectingServiceConfigurators.orElse(ImmutableList.of()),
                                           dependencyInjectors.orElse(ImmutableList.of()),
                                           beanFactory);

        return serverBuilder.build();
    }

    /**
     * Wrap {@link Server} with {@link SmartLifecycle}.
     */
    @Bean
    @ConditionalOnMissingBean(ArmeriaServerSmartLifecycle.class)
    public SmartLifecycle armeriaServerGracefulShutdownLifecycle(Server server) {
        return new ArmeriaServerGracefulShutdownLifecycle(server);
    }

    /**
     * Creates internal services that should not be exposed to the external network such as {@link DocService},
     * {@link PrometheusExpositionService} and {@link HealthCheckService}.
     *
     * 

Note that if a service path is either {@code null} or empty, the associated service will not be * initiated. For example, {@link ArmeriaSettings#getHealthCheckPath()} is {@code null}, * {@link HealthCheckService} will not be created automatically. * * @see ArmeriaSettings#getDocsPath() * @see ArmeriaSettings#getMetricsPath() * @see ArmeriaSettings#getHealthCheckPath() */ @Bean public InternalServices internalServices( ArmeriaSettings settings, Optional meterRegistry, Optional> healthCheckers, Optional> healthCheckServiceConfigurators, Optional> docServiceConfigurators, @Value("${management.server.port:#{null}}") @Nullable Integer managementServerPort) { return InternalServices.of(settings, meterRegistry.orElse(Metrics.globalRegistry), healthCheckers.orElse(ImmutableList.of()), healthCheckServiceConfigurators.orElse(ImmutableList.of()), docServiceConfigurators.orElse(ImmutableList.of()), managementServerPort); } /** * A user can configure a {@link Server} by providing an {@link ArmeriaServerConfigurator} bean. */ @Bean @ConditionalOnProperty("server.shutdown") public ArmeriaServerConfigurator gracefulShutdownServerConfigurator( @Value("${server.shutdown}") String shutdown, @Value("${spring.lifecycle.timeout-per-shutdown-phase:30s}") Duration duration) { if (GRACEFUL_SHUTDOWN.equalsIgnoreCase(shutdown)) { return sb -> sb.gracefulShutdownTimeout(duration, duration); } else { return sb -> {}; } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy