package io.spring.api; import io.restassured.module.mockmvc.RestAssuredMockMvc; import io.spring.JacksonCustomizations; import io.spring.api.security.WebSecurityConfig; import io.spring.application.CommentQueryService; import io.spring.application.data.CommentData; import io.spring.application.data.ProfileData; import io.spring.core.article.Article; import io.spring.core.article.ArticleRepository; import io.spring.core.comment.Comment; import io.spring.core.comment.CommentRepository; import io.spring.core.user.User; import org.junit.Before; import org.junit.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest; import org.springframework.boot.test.mock.mockito.MockBean; import org.springframework.context.annotation.Import; import org.springframework.test.web.servlet.MockMvc; import java.util.Arrays; import java.util.HashMap; import java.util.Map; import java.util.Optional; import static io.restassured.module.mockmvc.RestAssuredMockMvc.given; import static org.hamcrest.core.IsEqual.equalTo; import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.anyString; import static org.mockito.ArgumentMatchers.eq; import static org.mockito.Mockito.when; @WebMvcTest(CommentsApi.class) @Import({WebSecurityConfig.class, JacksonCustomizations.class}) public class CommentsApiTest extends TestWithCurrentUser { @MockBean private ArticleRepository articleRepository; @MockBean private CommentRepository commentRepository; @MockBean private CommentQueryService commentQueryService; private Article article; private CommentData commentData; private Comment comment; @Autowired private MockMvc mvc; @Before public void setUp() throws Exception { RestAssuredMockMvc.mockMvc(mvc); super.setUp(); article = new Article("title", "desc", "body", new String[]{"test", "java"}, user.getId()); when(articleRepository.findBySlug(eq(article.getSlug()))).thenReturn(Optional.of(article)); comment = new Comment("comment", user.getId(), article.getId()); commentData = new CommentData( comment.getId(), comment.getBody(), comment.getArticleId(), comment.getCreatedAt(), comment.getCreatedAt(), new ProfileData(user.getId(), user.getUsername(), user.getBio(), user.getImage(), false)); } @Test public void should_create_comment_success() throws Exception { Map param = new HashMap() {{ put("comment", new HashMap() {{ put("body", "comment content"); }}); }}; when(commentQueryService.findById(anyString(), eq(user))).thenReturn(Optional.of(commentData)); given() .contentType("application/json") .header("Authorization", "Token " + token) .body(param) .when() .post("/articles/{slug}/comments", article.getSlug()) .then() .statusCode(201) .body("comment.body", equalTo(commentData.getBody())); } @Test public void should_get_422_with_empty_body() throws Exception { Map param = new HashMap() {{ put("comment", new HashMap() {{ put("body", ""); }}); }}; given() .contentType("application/json") .header("Authorization", "Token " + token) .body(param) .when() .post("/articles/{slug}/comments", article.getSlug()) .then() .statusCode(422) .body("errors.body[0]", equalTo("can't be empty")); } @Test public void should_get_comments_of_article_success() throws Exception { when(commentQueryService.findByArticleId(anyString(), eq(null))).thenReturn(Arrays.asList(commentData)); RestAssuredMockMvc.when() .get("/articles/{slug}/comments", article.getSlug()) .prettyPeek() .then() .statusCode(200) .body("comments[0].id", equalTo(commentData.getId())); } @Test public void should_delete_comment_success() throws Exception { when(commentRepository.findById(eq(article.getId()), eq(comment.getId()))).thenReturn(Optional.of(comment)); given() .header("Authorization", "Token " + token) .when() .delete("/articles/{slug}/comments/{id}", article.getSlug(), comment.getId()) .then() .statusCode(204); } @Test public void should_get_403_if_not_author_of_article_or_author_of_comment_when_delete_comment() throws Exception { User anotherUser = new User("other@example.com", "other", "123", "", ""); when(userRepository.findByUsername(eq(anotherUser.getUsername()))).thenReturn(Optional.of(anotherUser)); when(jwtService.getSubFromToken(any())).thenReturn(Optional.of(anotherUser.getId())); when(userRepository.findById(eq(anotherUser.getId()))).thenReturn(Optional.ofNullable(anotherUser)); when(commentRepository.findById(eq(article.getId()), eq(comment.getId()))).thenReturn(Optional.of(comment)); String token = jwtService.toToken(anotherUser); when(userRepository.findById(eq(anotherUser.getId()))).thenReturn(Optional.of(anotherUser)); given() .header("Authorization", "Token " + token) .when() .delete("/articles/{slug}/comments/{id}", article.getSlug(), comment.getId()) .then() .statusCode(403); } }