Spring boot 2.04 Джексон не может сериализовать LocalDateTime в строку - PullRequest
0 голосов
/ 11 сентября 2018

У меня есть загрузочное приложение Spring и свойства LocalDateTime, сериализованные как массив JSON, а не String.

Как я обнаружил при поиске в Интернете, все, что мне нужно, это настроить в application.properties

spring.jackson.serialization.write-dates-as-timestamps=false

Я также пытался поместить jackson-datatype-jsr310 в зависимости, но тоже не повезло

Также я попытался добавить аннотацию:

@DateTimeFormat(iso = ISO.DATE_TIME)

это тоже не помогает.

Я видел, что многие люди сталкивались с чем-то похожим, но их решения, похоже, связаны с Spring Boot 1.x, а я использую 2.04

Также я использую Lombok, но не уверен, что это влияет на формат сериализации.

Как я могу отследить и исправить формат сериализации даты, чтобы она была строкой даты ISO?

Вот пример ответа (начало - LocalDateTime, и я хочу, чтобы оно было в виде строки ISO):

{
  "id": 3,
  "enabled": true,
  "outletId": 5,
  "reason": "hello",
  "start": [
    2019,
    9,
    10,
    10,
    42,
    11
  ],
  "status": "AVAILABLE"
}

Здесь объект ответа метода REST-контроллера:

@Data
@Entity
@Table(indexes = { @Index(columnList = ("outletId"),name="outlet_id_index"), 
        @Index(columnList = ("start"),name="start_index"),
        @Index(columnList = ("outletId, start"),name="outlet_id_start_index")})
public class OutletChron extends BaseEntity {
    private Long outletId;
    private String reason;

    @DateTimeFormat(iso = ISO.DATE_TIME)
    private LocalDateTime start;
    @Enumerated(EnumType.STRING)
    @Column(length = 30)
    private OutletStatus status;
}

Вот мой POM:

<?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 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.banquets</groupId>
    <artifactId>Banquet</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <packaging>jar</packaging>

    <name>Banquet</name>
    <description></description>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.0.4.RELEASE</version>
        <relativePath /> <!-- lookup parent from repository -->
    </parent>

    <properties>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
        <java.version>1.8</java.version>
        <spring-cloud.version>Finchley.SR1</spring-cloud.version>
    </properties>

    <dependencies>
        <dependency>
            <groupId>com.fasterxml.jackson.datatype</groupId>
            <artifactId>jackson-datatype-jsr310</artifactId>
        </dependency>
        <dependency>
            <groupId>io.springfox</groupId>
            <artifactId>springfox-swagger2</artifactId>
            <version>2.9.2</version>
        </dependency>
        <dependency>
            <groupId>io.springfox</groupId>
            <artifactId>springfox-swagger-ui</artifactId>
            <version>2.9.2</version>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</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-web</artifactId>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-devtools</artifactId>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
        </dependency>

        <!-- <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web-services</artifactId> 
            </dependency> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId> 
            </dependency> -->

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
    </dependencies>

    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>${spring-cloud.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

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


</project>

UPDATE

Я строю новый проект только для тестирования, и я обнаружил, что Формат строки был по умолчанию для отображения LocalDateTime . Я смог отследить эти изменения формата после настройки swagger. Так что без этого чванства у меня есть строковый формат:

@Configuration
@EnableSwagger2
public class SwaggerConfig extends WebMvcConfigurationSupport {

    @Autowired
    ServletContext servletContext;

    @Bean
    public Docket productApi() {
        return new Docket(DocumentationType.SWAGGER_2)
                .select()
                .apis(RequestHandlerSelectors.basePackage("com.demo"))
                .build();
    }

    @Override
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
        registry.addResourceHandler("swagger-ui.html")
                .addResourceLocations("classpath:/META-INF/resources/");

        registry.addResourceHandler("/webjars/**")
                .addResourceLocations("classpath:/META-INF/resources/webjars/");
    }
}

ОБНОВЛЕНИЕ эта конфигурация Swagger, кажется, работает (формат даты - String, и я могу получить доступ к интерфейсу Swagger в http://localhost:8000/api/swagger-ui.html

@Configuration
@EnableSwagger2
public class SwaggerConfig {

    @Bean
    public Docket apiDocket() {
        return new Docket(DocumentationType.SWAGGER_2)
                .select()
                .apis(RequestHandlerSelectors.any())
                .paths(PathSelectors.any())
                .build();
    }
}

Ответы [ 3 ]

0 голосов
/ 11 сентября 2018

Для меня следующие работы:

@JsonFormat(pattern = "dd.MM.yyyy HH:mm")
private LocalDateTime startTime;

Это напечатает дату в строковом формате, например как 11.09.2018 15: 44

0 голосов
/ 11 сентября 2018

Spring boot 2.x не требует импорта спецификаций JSR310, поскольку теперь он входит в состав Spring Framework, поэтому нет необходимости импортировать их, и форматирование строк будет работать автоматически.

Если вам нужно переопределить некоторые настройки по умолчаниювесенней загрузки, тогда вам нужно реализовать WebMvcConfigurer вместо расширения WebMvcConfigurationSupport.

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

Если ресурсы в пути по умолчанию не требуются, и простое удаление расширения WebMvcConfigurationSupport будет работать для форматирования строки по умолчанию.

ОБНОВЛЕННЫЙ ОТВЕТ:

Если вы используете WebMvcConfigurationSupport, это означает, что он не указывает на автоматическую настройку Spring MVC, означает, что настройки по умолчанию не будут работать, и вам придется все здесь определять, переопределяя его методы поддержки.Так что вместо WebMvcConfigurationSupport внедряйте WebMvcConfigurer вместо.

Вот обновленный конфиг.

@Configuration
@EnableSwagger2
public class SwaggerConfig implements WebMvcConfigurer {

    @Bean
    public Docket productApi() {
        return new Docket(DocumentationType.SWAGGER_2)
                .select()
                .apis(RequestHandlerSelectors.basePackage("com.demo"))
                .build();
    }

    @Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {
        registry.addResourceHandler("swagger-ui.html")
                .addResourceLocations("classpath:/META-INF/resources/");

        registry.addResourceHandler("/webjars/**")
                .addResourceLocations("classpath:/META-INF/resources/webjars/");
    }
}
0 голосов
/ 11 сентября 2018

Создать объект сопоставления вручную

@Bean
@Primary
public ObjectMapper objectMapper() {
    ObjectMapper mapper = new ObjectMapper();

    JavaTimeModule timeModule = new JavaTimeModule();
    mapper.registerModule(timeModule);

    mapper.configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false);
    mapper.configure(SerializationFeature.WRITE_DATE_TIMESTAMPS_AS_NANOSECONDS, false);

    return mapper;
}
...