Spring Boot: HATEOAS и пользовательский JacksonObjectMapper - PullRequest
0 голосов
/ 10 марта 2020

После того, как я добавил зависимость для HATEOAS в Maven, Spring Boot не запускается:

Добавлена ​​зависимость:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-hateoas</artifactId>
</dependency>

Full pom. xml:

<?xml version="1.0" encoding="UTF-8"?>
    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.5.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>ru.example</groupId>
    <artifactId>testapp</artifactId>
    <version>1.0</version>
    <name>testapp</nAfter I added dependency for HATEOAS to Maven, Spring Boot does not startame>
    <description>Test</description>

    <properties>
        <java.version>1.8</java.version>
        <h2.version>1.4.200</h2.version>
        <jackson-json.version>2.10.2</jackson-json.version>
        <jsoup.version>1.12.1</jsoup.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-jpa</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-hateoas</artifactId>
        </dependency>
        <dependency>
            <groupId>org.hibernate.validator</groupId>
            <artifactId>hibernate-validator</artifactId>
            <version>6.0.17.Final</version>
        </dependency>
        <dependency>
            <groupId>org.jsoup</groupId>
            <artifactId>jsoup</artifactId>
            <version>${jsoup.version}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-security</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.security</groupId>
            <artifactId>spring-security-test</artifactId>
            <version>5.3.0.RELEASE</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>

        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>

        <dependency>
            <groupId>com.h2database</groupId>
            <artifactId>h2</artifactId>
            <version>${h2.version}</version>
        </dependency>

        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
            <version>${jackson-json.version}</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.datatype</groupId>
            <artifactId>jackson-datatype-hibernate5</artifactId>
            <version>${jackson-json.version}</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.datatype</groupId>
            <artifactId>jackson-datatype-jsr310</artifactId>
            <version>${jackson-json.version}</version>
        </dependency>

    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

SecurityConfig.class:

package ru.example.testapp;

import lombok.RequiredArgsConstructor;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Configuration;
import org.springframework.security.config.annotation.authentication.builders.AuthenticationManagerBuilder;
import org.springframework.security.config.annotation.web.builders.HttpSecurity;
import org.springframework.security.config.annotation.web.configuration.WebSecurityConfigurerAdapter;
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
import ru.example.testapp.dao.UserRepository;
import ru.example.testapp.service.UserServiceImpl;

@Configuration
@RequiredArgsConstructor
public class SecurityConfig extends WebSecurityConfigurerAdapter {
    private final UserRepository userRepository;

    @Autowired
    public void configureGlobal(AuthenticationManagerBuilder auth) throws Exception {
        auth
                .userDetailsService(new UserServiceImpl(userRepository))
                .passwordEncoder(new BCryptPasswordEncoder());
    }

    protected void configure(final HttpSecurity http) throws Exception {
        http
                .authorizeRequests()
                .antMatchers("/rest/admin/**").hasRole("ADMIN").and().httpBasic().and()
                .authorizeRequests()
                .antMatchers("/rest/user/**").hasAnyRole("USER","ADMIN").and().httpBasic().and()
                .authorizeRequests().and()
                .csrf().ignoringAntMatchers("/rest/**");
    }
}

JacksonObjectMapper.class:

package ru.example.testapp.util.json;

import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.PropertyAccessor;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.datatype.hibernate5.Hibernate5Module;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import org.springframework.stereotype.Component;

@Component
public class JacksonObjectMapper extends ObjectMapper {

    private static final ObjectMapper MAPPER = new JacksonObjectMapper();

    public static ObjectMapper getMapper() {
        return MAPPER;
    }

    private JacksonObjectMapper() {
        registerModule(new Hibernate5Module());
        registerModule(new JavaTimeModule());
        configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
        configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false);
        setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.NONE);
        setVisibility(PropertyAccessor.FIELD, JsonAutoDetect.Visibility.ANY);
        setSerializationInclusion(JsonInclude.Include.NON_NULL);
    }
}

В консоли возникла следующая ошибка:

org.springframework.beans. factory.UnsatisfiedDependencyException: Ошибка создания компонента с именем 'securityConfig': Неудовлетворенная зависимость, выраженная через метод 'setContentNegotationStrategy', параметр 0; вложенное исключение: org.springframework.beans.factory.UnsatisfiedDependencyException: ошибка при создании bean-компонента с именем org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration $ EnableWebMvcConfiguration 'параметром метода 0; Вложенное исключение - org.springframework.beans.factory.UnsatisfiedDependencyException: ошибка создания бина с именем 'hypermediaWebMvcConfigurer', определенным в ресурсе пути к классу [org / springframework / hateoas / config / WebMvcHateoasConfiguration.class]: выраженный параметр -media 0 ; вложенным исключением является org.springframework.beans.factory.BeanCreationException: ошибка при создании bean-компонента с именем 'hypermediaWebMvcConverters', определенного в ресурсе пути к классу [org / springframework / hateoas / config / HateoasConfiguration.class]: сбой создания экземпляра бина через метод фабрики; вложенное исключение: org.springframework.beans.BeanInstantiationException: не удалось создать экземпляр [org.springframework.hateoas.config.WebConverters]: метод фабрики 'hypermediaWebMvcConverters' вызвал исключение; вложенное исключение: java .lang.IllegalStateException: сбой copy (): ru.example.wmanage.util. json .JacksonObjectMapper (версия: 2.10.2) не переопределяет copy (); это должно

В чем может быть проблема? По ошибке - что-то с securityConfig, JacksonObjectMapper и hateoas. Если я удаляю spring-boot-starter-hateoas при зависимостях, то все работает. Но мне нужны хатеоас. Пожалуйста, помогите.

ОБНОВЛЕНО: проблема возникает при использовании пользовательского JacksonObjectMapper с аннотацией @Component. Как только spring-boot-starter-hateoas добавлен в зависимости, Spting Boot не запускается.

ВОПРОС: Как использовать пользовательские JacksonObjectMapper и hateoas вместе?

ПРОБЛЕМА НЕ РАЗРЕШЕНА

1 Ответ

1 голос
/ 10 марта 2020

Я не думаю, что у вас есть основания расширять ObjectMapper. Вы должны создать экземпляр ObjectMapper, как обычно, а затем настроить его с помощью его открытых методов и зарегистрировать его как компонент в вашей конфигурации.

@Bean
public ObjectMapper createMapper() {
    return new ObjectMapper().registerModule(new Hibernate5Module());
            .registerModule(new JavaTimeModule());
            .configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
            .configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false);
            .setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.NONE);
            .setVisibility(PropertyAccessor.FIELD, JsonAutoDetect.Visibility.ANY);
            .setSerializationInclusion(JsonInclude.Include.NON_NULL);
}
...