Use Instant in DTO

This commit is contained in:
René Pfeuffer
2018-06-01 14:19:12 +02:00
parent cb4f2e49d9
commit c62168274d
7 changed files with 66 additions and 12 deletions

View File

@@ -107,7 +107,16 @@
<artifactId>jackson-jaxrs-base</artifactId> <artifactId>jackson-jaxrs-base</artifactId>
<version>${jackson.version}</version> <version>${jackson.version}</version>
</dependency> </dependency>
<dependency>
<groupId>com.fasterxml.jackson.datatype</groupId>
<artifactId>jackson-datatype-jdk8</artifactId>
<version>${jackson.version}</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.datatype</groupId>
<artifactId>jackson-datatype-jsr310</artifactId>
<version>${jackson.version}</version>
</dependency>
<!-- rest api --> <!-- rest api -->
<dependency> <dependency>

View File

@@ -36,7 +36,10 @@ import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.introspect.AnnotationIntrospectorPair; import com.fasterxml.jackson.databind.introspect.AnnotationIntrospectorPair;
import com.fasterxml.jackson.databind.introspect.JacksonAnnotationIntrospector; import com.fasterxml.jackson.databind.introspect.JacksonAnnotationIntrospector;
import com.fasterxml.jackson.databind.type.TypeFactory; import com.fasterxml.jackson.databind.type.TypeFactory;
import com.fasterxml.jackson.datatype.jdk8.Jdk8Module;
import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule;
import com.fasterxml.jackson.module.jaxb.JaxbAnnotationIntrospector; import com.fasterxml.jackson.module.jaxb.JaxbAnnotationIntrospector;
import javax.ws.rs.Produces; import javax.ws.rs.Produces;
import javax.ws.rs.core.MediaType; import javax.ws.rs.core.MediaType;
import javax.ws.rs.ext.ContextResolver; import javax.ws.rs.ext.ContextResolver;
@@ -55,7 +58,9 @@ public final class JSONContextResolver implements ContextResolver<ObjectMapper>
private final ObjectMapper mapper; private final ObjectMapper mapper;
public JSONContextResolver() { public JSONContextResolver() {
mapper = new ObjectMapper(); mapper = new ObjectMapper()
.registerModule(new Jdk8Module())
.registerModule(new JavaTimeModule());
mapper.setAnnotationIntrospector(createAnnotationIntrospector()); mapper.setAnnotationIntrospector(createAnnotationIntrospector());
mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); mapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
} }

View File

@@ -1,15 +1,13 @@
package sonia.scm.api.v2.resources; package sonia.scm.api.v2.resources;
import org.apache.shiro.SecurityUtils; import org.apache.shiro.SecurityUtils;
import org.mapstruct.AfterMapping; import org.mapstruct.*;
import org.mapstruct.Context;
import org.mapstruct.Mapper;
import org.mapstruct.MappingTarget;
import sonia.scm.api.rest.resources.UserResource; import sonia.scm.api.rest.resources.UserResource;
import sonia.scm.security.Role; import sonia.scm.security.Role;
import sonia.scm.user.User; import sonia.scm.user.User;
import javax.ws.rs.core.UriInfo; import javax.ws.rs.core.UriInfo;
import java.time.Instant;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
@@ -36,4 +34,9 @@ public abstract class User2UserDtoMapper {
} }
target.setLinks(links); target.setLinks(links);
} }
@Mappings({@Mapping(target = "lastModified"), @Mapping(target = "creationDate")})
Instant mapTime(Long epochMilli) {
return epochMilli == null? null: Instant.ofEpochMilli(epochMilli);
}
} }

View File

@@ -3,15 +3,16 @@ package sonia.scm.api.v2.resources;
import lombok.Data; import lombok.Data;
import javax.xml.bind.annotation.XmlElement; import javax.xml.bind.annotation.XmlElement;
import java.time.Instant;
import java.util.Map; import java.util.Map;
@Data @Data
public class UserDto { public class UserDto {
private boolean active; private boolean active;
private boolean admin; private boolean admin;
private Long creationDate; private Instant creationDate;
private String displayName; private String displayName;
private Long lastModified; private Instant lastModified;
private String mail; private String mail;
private String name; private String name;
private String password; private String password;

View File

@@ -2,12 +2,11 @@ package sonia.scm.api.v2.resources;
import com.google.inject.Inject; import com.google.inject.Inject;
import org.apache.shiro.authc.credential.PasswordService; import org.apache.shiro.authc.credential.PasswordService;
import org.mapstruct.Context; import org.mapstruct.*;
import org.mapstruct.Mapper;
import org.mapstruct.Mapping;
import org.mapstruct.Named;
import sonia.scm.user.User; import sonia.scm.user.User;
import java.time.Instant;
import static sonia.scm.api.rest.resources.UserResource.DUMMY_PASSWORT; import static sonia.scm.api.rest.resources.UserResource.DUMMY_PASSWORT;
@Mapper @Mapper
@@ -30,6 +29,10 @@ public abstract class UserDto2UserMapper {
{ {
return passwordService.encryptPassword(password); return passwordService.encryptPassword(password);
} }
}
@Mappings({@Mapping(target = "lastModified"), @Mapping(target = "creationDate")})
Long mapTime(Instant instant) {
return instant == null? null: instant.toEpochMilli();
} }
} }

View File

@@ -12,6 +12,7 @@ import sonia.scm.user.User;
import javax.ws.rs.core.UriInfo; import javax.ws.rs.core.UriInfo;
import java.net.URI; import java.net.URI;
import java.net.URISyntaxException; import java.net.URISyntaxException;
import java.time.Instant;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNull; import static org.junit.Assert.assertNull;
@@ -84,4 +85,19 @@ public class User2UserDtoMapperTest {
assertEquals(UserResource.DUMMY_PASSWORT, userDto.getPassword()); assertEquals(UserResource.DUMMY_PASSWORT, userDto.getPassword());
} }
@Test
public void shouldMapTimes() {
User user = new User();
user.setName("abc");
Instant expectedCreationDate = Instant.ofEpochSecond(6666666);
Instant expectedModificationDate = expectedCreationDate.plusSeconds(1);
user.setCreationDate(expectedCreationDate.toEpochMilli());
user.setLastModified(expectedModificationDate.toEpochMilli());
UserDto userDto = mapper.userToUserDto(user, uriInfo);
assertEquals(expectedCreationDate, userDto.getCreationDate());
assertEquals(expectedModificationDate, userDto.getLastModified());
}
} }

View File

@@ -7,6 +7,8 @@ import org.mockito.InjectMocks;
import org.mockito.Mock; import org.mockito.Mock;
import sonia.scm.user.User; import sonia.scm.user.User;
import java.time.Instant;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
import static org.mockito.Mockito.when; import static org.mockito.Mockito.when;
import static org.mockito.MockitoAnnotations.initMocks; import static org.mockito.MockitoAnnotations.initMocks;
@@ -36,6 +38,21 @@ public class UserDto2UserMapperTest {
assertEquals("encrypted" , user.getPassword()); assertEquals("encrypted" , user.getPassword());
} }
@Test
public void shouldMapTimes() {
UserDto dto = new UserDto();
dto.setName("abc");
Instant expectedCreationDate = Instant.ofEpochMilli(66666660000L);
Instant expectedModificationDate = null;
dto.setCreationDate(expectedCreationDate);
dto.setLastModified(expectedModificationDate);
User user = mapper.userDtoToUser(dto, "original password");
assertEquals((Long) expectedCreationDate.toEpochMilli(), user.getCreationDate());
assertEquals(null, user.getLastModified());
}
@Before @Before
public void init() { public void init() {
initMocks(this); initMocks(this);