blob: 5f0cd986897db380f95b6663229a1703a32c74c6 [file] [log] [blame]
/*
* Copyright (c) 2017, 2021 Oracle and/or its affiliates. All rights reserved.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0, which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception, which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
*/
package org.glassfish.jersey.server;
import java.util.Collection;
import java.util.Collections;
import java.util.HashSet;
import java.util.IdentityHashMap;
import java.util.Set;
import java.util.function.BiPredicate;
import java.util.logging.Logger;
import java.util.stream.Collectors;
import javax.ws.rs.RuntimeType;
import org.glassfish.jersey.internal.BootstrapBag;
import org.glassfish.jersey.internal.BootstrapConfigurator;
import org.glassfish.jersey.internal.inject.InjectionManager;
import org.glassfish.jersey.internal.inject.ProviderBinder;
import org.glassfish.jersey.internal.inject.Providers;
import org.glassfish.jersey.model.ContractProvider;
import org.glassfish.jersey.model.internal.ComponentBag;
import org.glassfish.jersey.server.internal.JerseyResourceContext;
import org.glassfish.jersey.server.internal.LocalizationMessages;
import org.glassfish.jersey.server.model.ModelProcessor;
import org.glassfish.jersey.server.model.Resource;
import org.glassfish.jersey.server.model.ResourceModel;
import org.glassfish.jersey.server.spi.ComponentProvider;
/**
* Configurator which binds providers and resources into {@link InjectionManager}.
*
* @author Petr Bouda
*/
public class ResourceModelConfigurator implements BootstrapConfigurator {
private static final Logger LOGGER = Logger.getLogger(ResourceModelConfigurator.class.getName());
@Override
public void init(InjectionManager injectionManager, BootstrapBag bootstrapBag) {
ServerBootstrapBag serverBag = (ServerBootstrapBag) bootstrapBag;
Collection<ModelProcessor> modelProcessors = serverBag.getModelProcessors();
ResourceConfig runtimeConfig = serverBag.getRuntimeConfig();
ResourceBag resourceBag = serverBag.getResourceBag();
ComponentBag componentBag = runtimeConfig.getComponentBag();
// Adds all providers from resource config to InjectionManager -> BootstrapConfigurators are able to work with these
// services and get them.
bindProvidersAndResources(
injectionManager, serverBag, componentBag, resourceBag.classes, resourceBag.instances, runtimeConfig);
ResourceModel resourceModel = new ResourceModel.Builder(resourceBag.getRootResources(), false).build();
resourceModel = processResourceModel(modelProcessors, resourceModel, runtimeConfig);
bindEnhancingResourceClasses(injectionManager, serverBag, resourceModel, resourceBag, runtimeConfig);
serverBag.setResourceModel(resourceModel);
// Add newly created resource model in ResourceContext.
serverBag.getResourceContext().setResourceModel(resourceModel);
}
private ResourceModel processResourceModel(Collection<ModelProcessor> modelProcessors, ResourceModel resourceModel,
ResourceConfig runtimeConfig) {
for (final ModelProcessor modelProcessor : modelProcessors) {
resourceModel = modelProcessor.processResourceModel(resourceModel, runtimeConfig);
}
return resourceModel;
}
private void bindEnhancingResourceClasses(
InjectionManager injectionManager,
ServerBootstrapBag bootstrapBag,
ResourceModel resourceModel,
ResourceBag resourceBag,
ResourceConfig runtimeConfig) {
Set<Class<?>> newClasses = new HashSet<>();
Set<Object> newInstances = new HashSet<>();
for (final Resource res : resourceModel.getRootResources()) {
newClasses.addAll(res.getHandlerClasses());
newInstances.addAll(res.getHandlerInstances());
}
newClasses.removeAll(resourceBag.classes);
newInstances.removeAll(resourceBag.instances);
ComponentBag emptyComponentBag = ComponentBag.newInstance(input -> false);
bindProvidersAndResources(injectionManager, bootstrapBag, emptyComponentBag, newClasses, newInstances, runtimeConfig);
}
private void bindProvidersAndResources(
InjectionManager injectionManager,
ServerBootstrapBag bootstrapBag,
ComponentBag componentBag,
Collection<Class<?>> resourceClasses,
Collection<Object> resourceInstances,
ResourceConfig runtimeConfig) {
Collection<ComponentProvider> componentProviders = bootstrapBag.getComponentProviders().get();
JerseyResourceContext resourceContext = bootstrapBag.getResourceContext();
Set<Class<?>> registeredClasses = runtimeConfig.getRegisteredClasses();
/*
* Check the {@code component} whether it is correctly configured for client or server {@link RuntimeType runtime}.
*/
java.util.function.Predicate<Class<?>> correctlyConfigured =
componentClass -> Providers.checkProviderRuntime(componentClass,
componentBag.getModel(componentClass),
RuntimeType.SERVER,
!registeredClasses.contains(componentClass),
resourceClasses.contains(componentClass));
/*
* Check the {@code resource class} whether it is correctly configured for client or server {@link RuntimeType runtime}.
*/
BiPredicate<Class<?>, ContractProvider> correctlyConfiguredResource =
(resourceClass, model) -> Providers.checkProviderRuntime(
resourceClass,
model,
RuntimeType.SERVER,
!registeredClasses.contains(resourceClass),
true);
Set<Class<?>> componentClasses =
componentBag.getClasses(ComponentBag.excludeMetaProviders(injectionManager)).stream()
.filter(correctlyConfigured)
.collect(Collectors.toSet());
// Merge programmatic resource classes with component classes.
Set<Class<?>> classes = Collections.newSetFromMap(new IdentityHashMap<>());
classes.addAll(componentClasses);
classes.addAll(resourceClasses);
// Bind classes.
for (final Class<?> componentClass: classes) {
ContractProvider model = componentBag.getModel(componentClass);
if (bindWithComponentProvider(componentClass, model, componentProviders)) {
continue;
}
if (resourceClasses.contains(componentClass)) {
if (!Resource.isAcceptable(componentClass)) {
LOGGER.warning(LocalizationMessages.NON_INSTANTIABLE_COMPONENT(componentClass));
continue;
}
if (model != null && !correctlyConfiguredResource.test(componentClass, model)) {
model = null;
}
resourceContext.unsafeBindResource(componentClass, model);
} else {
ProviderBinder.bindProvider(componentClass, model, injectionManager);
}
}
// Merge programmatic resource instances with other component instances.
Set<Object> instances =
componentBag.getInstances(ComponentBag.excludeMetaProviders(injectionManager)).stream()
.filter(instance -> correctlyConfigured.test(instance.getClass()))
.collect(Collectors.toSet());
instances.addAll(resourceInstances);
// Bind instances.
for (Object component: instances) {
ContractProvider model = componentBag.getModel(component.getClass());
if (resourceInstances.contains(component)) {
if (model != null && !correctlyConfiguredResource.test(component.getClass(), model)) {
model = null;
}
resourceContext.unsafeBindResource(component, model);
} else {
ProviderBinder.bindProvider(component, model, injectionManager);
}
}
}
private boolean bindWithComponentProvider(
Class<?> component, ContractProvider providerModel, Iterable<ComponentProvider> componentProviders) {
for (ComponentProvider provider : componentProviders) {
if (provider.bind(component, providerModel)) {
return true;
}
}
return false;
}
}