org.springframework.data.rest.webmvc.config.PersistentEntityResourceHandlerMethodArgumentResolver.java Source code

Java tutorial

Introduction

Here is the source code for org.springframework.data.rest.webmvc.config.PersistentEntityResourceHandlerMethodArgumentResolver.java

Source

/*
 * Copyright 2012-2019 the original author or authors.
 *
 * 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
 *
 *      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 org.springframework.data.rest.webmvc.config;

import lombok.NonNull;
import lombok.RequiredArgsConstructor;

import java.io.IOException;
import java.io.Serializable;
import java.util.List;
import java.util.Optional;

import javax.servlet.http.HttpServletRequest;

import org.springframework.core.MethodParameter;
import org.springframework.core.convert.ConversionService;
import org.springframework.core.convert.support.DefaultConversionService;
import org.springframework.data.mapping.IdentifierAccessor;
import org.springframework.data.mapping.PersistentEntity;
import org.springframework.data.mapping.PersistentProperty;
import org.springframework.data.mapping.PersistentPropertyAccessor;
import org.springframework.data.mapping.model.ConvertingPropertyAccessor;
import org.springframework.data.rest.core.support.EntityLookup;
import org.springframework.data.rest.webmvc.IncomingRequest;
import org.springframework.data.rest.webmvc.PersistentEntityResource;
import org.springframework.data.rest.webmvc.PersistentEntityResource.Builder;
import org.springframework.data.rest.webmvc.ResourceNotFoundException;
import org.springframework.data.rest.webmvc.RootResourceInformation;
import org.springframework.data.rest.webmvc.json.DomainObjectReader;
import org.springframework.data.rest.webmvc.support.BackendIdHandlerMethodArgumentResolver;
import org.springframework.http.MediaType;
import org.springframework.http.converter.HttpMessageConverter;
import org.springframework.http.converter.HttpMessageNotReadableException;
import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServletServerHttpRequest;
import org.springframework.plugin.core.PluginRegistry;
import org.springframework.web.bind.support.WebDataBinderFactory;
import org.springframework.web.context.request.NativeWebRequest;
import org.springframework.web.method.support.HandlerMethodArgumentResolver;
import org.springframework.web.method.support.ModelAndViewContainer;

import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ObjectNode;

/**
 * Custom {@link HandlerMethodArgumentResolver} to create {@link PersistentEntityResource} instances.
 *
 * @author Jon Brisbin
 * @author Oliver Gierke
 */
@RequiredArgsConstructor
public class PersistentEntityResourceHandlerMethodArgumentResolver implements HandlerMethodArgumentResolver {

    private static final String ERROR_MESSAGE = "Could not read an object of type %s from the request!";
    private static final String NO_CONVERTER_FOUND = "No suitable HttpMessageConverter found to read request body into object of type %s from request with content type of %s!";

    private final @NonNull List<HttpMessageConverter<?>> messageConverters;
    private final @NonNull RootResourceInformationHandlerMethodArgumentResolver resourceInformationResolver;
    private final @NonNull BackendIdHandlerMethodArgumentResolver idResolver;
    private final @NonNull DomainObjectReader reader;
    private final @NonNull PluginRegistry<EntityLookup<?>, Class<?>> lookups;
    private final ConversionService conversionService = new DefaultConversionService();

    /*
     * (non-Javadoc)
     * @see org.springframework.web.method.support.HandlerMethodArgumentResolver#supportsParameter(org.springframework.core.MethodParameter)
     */
    @Override
    public boolean supportsParameter(MethodParameter parameter) {
        return PersistentEntityResource.class.isAssignableFrom(parameter.getParameterType());
    }

    /*
     * (non-Javadoc)
     * @see org.springframework.web.method.support.HandlerMethodArgumentResolver#resolveArgument(org.springframework.core.MethodParameter, org.springframework.web.method.support.ModelAndViewContainer, org.springframework.web.context.request.NativeWebRequest, org.springframework.web.bind.support.WebDataBinderFactory)
     */
    @Override
    @SuppressWarnings({ "unchecked", "rawtypes" })
    public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer,
            NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception {

        RootResourceInformation resourceInformation = resourceInformationResolver.resolveArgument(parameter,
                mavContainer, webRequest, binderFactory);

        HttpServletRequest nativeRequest = webRequest.getNativeRequest(HttpServletRequest.class);
        ServletServerHttpRequest request = new ServletServerHttpRequest(nativeRequest);
        IncomingRequest incoming = new IncomingRequest(request);

        Class<?> domainType = resourceInformation.getDomainType();
        MediaType contentType = request.getHeaders().getContentType();

        for (HttpMessageConverter converter : messageConverters) {

            if (!converter.canRead(PersistentEntityResource.class, contentType)) {
                continue;
            }

            Optional<Serializable> id = Optional
                    .ofNullable(idResolver.resolveArgument(parameter, mavContainer, webRequest, binderFactory));
            Optional<Object> objectToUpdate = id.flatMap(it -> resourceInformation.getInvoker().invokeFindById(it));
            Object newObject = read(resourceInformation, incoming, converter, objectToUpdate);

            if (newObject == null) {
                throw new HttpMessageNotReadableException(String.format(ERROR_MESSAGE, domainType), request);
            }

            PersistentEntity<?, ?> entity = resourceInformation.getPersistentEntity();

            if (!id.isPresent()) {
                return toResource(newObject, entity, false);
            }

            PersistentPropertyAccessor<Object> accessor = entity.getPropertyAccessor(newObject);
            PersistentProperty<?> idProperty = entity.getRequiredIdProperty();

            boolean forUpdate = objectToUpdate.isPresent();

            // Transfer identifier from existing object
            objectToUpdate.map(entity::getIdentifierAccessor) //
                    .map(IdentifierAccessor::getIdentifier) //
                    .ifPresent(it -> accessor.setProperty(idProperty, it));

            if (!forUpdate) {

                // Find property to map URI derived value from
                PersistentProperty<?> propertyToSet = lookups.getPluginFor(domainType) //
                        .flatMap(EntityLookup::getLookupProperty) //
                        .<PersistentProperty<?>>map(entity::getPersistentProperty) //
                        .orElseGet(() -> idProperty);

                // Transfer onto new object
                new ConvertingPropertyAccessor(accessor, conversionService) //
                        .setProperty(propertyToSet, id.get());
            }

            return toResource(accessor.getBean(), entity, forUpdate);
        }

        throw new HttpMessageNotReadableException(String.format(NO_CONVERTER_FOUND, domainType, contentType),
                request);
    }

    /**
     * Reads the given {@link ServerHttpRequest} into an object of the type of the given {@link RootResourceInformation},
     * potentially applying the content to an object of the given id.
     *
     * @param information must not be {@literal null}.
     * @param request must not be {@literal null}.
     * @param converter must not be {@literal null}.
     * @param id must not be {@literal null}.
     * @return
     */
    private Object read(RootResourceInformation information, IncomingRequest request,
            HttpMessageConverter<Object> converter, Optional<Object> objectToUpdate) {

        // JSON + PATCH request
        if (request.isPatchRequest() && converter instanceof MappingJackson2HttpMessageConverter) {

            return objectToUpdate.map(it -> {

                ObjectMapper mapper = ((MappingJackson2HttpMessageConverter) converter).getObjectMapper();
                return readPatch(request, mapper, it);

            }).orElseThrow(() -> new ResourceNotFoundException());

            // JSON + PUT request
        } else if (converter instanceof MappingJackson2HttpMessageConverter) {

            ObjectMapper mapper = ((MappingJackson2HttpMessageConverter) converter).getObjectMapper();

            return objectToUpdate.map(it -> readPutForUpdate(request, mapper, it))//
                    .orElseGet(() -> read(request, converter, information));
        }

        // Catch all
        return read(request, converter, information);
    }

    private Object readPatch(IncomingRequest request, ObjectMapper mapper, Object existingObject) {

        try {

            JsonPatchHandler handler = new JsonPatchHandler(mapper, reader);
            return handler.apply(request, existingObject);

        } catch (Exception o_O) {

            if (o_O instanceof HttpMessageNotReadableException) {
                throw (HttpMessageNotReadableException) o_O;
            }

            throw new HttpMessageNotReadableException(String.format(ERROR_MESSAGE, existingObject.getClass()), o_O,
                    request.getServerHttpRequest());
        }
    }

    private Object readPutForUpdate(IncomingRequest request, ObjectMapper mapper, Object existingObject) {

        try {

            JsonPatchHandler handler = new JsonPatchHandler(mapper, reader);
            JsonNode jsonNode = mapper.readTree(request.getBody());

            return handler.applyPut((ObjectNode) jsonNode, existingObject);

        } catch (Exception o_O) {
            throw new HttpMessageNotReadableException(String.format(ERROR_MESSAGE, existingObject.getClass()), o_O,
                    request.getServerHttpRequest());
        }
    }

    private Object read(IncomingRequest request, HttpMessageConverter<Object> converter,
            RootResourceInformation information) {

        try {
            return converter.read(information.getDomainType(), request.getServerHttpRequest());
        } catch (IOException o_O) {
            throw new HttpMessageNotReadableException(String.format(ERROR_MESSAGE, information.getDomainType()),
                    o_O, request.getServerHttpRequest());
        }
    }

    private PersistentEntityResource toResource(Object bean, PersistentEntity<?, ?> entity, boolean forUpdate) {

        Builder build = PersistentEntityResource.build(bean, entity);
        return forUpdate ? build.build() : build.forCreation();
    }
}