From 628df100287ba677606aaee7fcd2291d144702dd Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Wed, 1 May 2024 20:26:49 +0700 Subject: [PATCH 01/20] [FIX] Fix CD workflow --- .github/workflows/cd.yml | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/.github/workflows/cd.yml b/.github/workflows/cd.yml index 47d1fb0..ebdb71d 100644 --- a/.github/workflows/cd.yml +++ b/.github/workflows/cd.yml @@ -76,6 +76,16 @@ jobs: name: Publish Docker Image runs-on: ubuntu-latest needs: test + services: + postgres: + image: postgres:latest + env: + POSTGRES_USER: postgres + POSTGRES_PASSWORD: postgres + POSTGRES_DB: snackscription_review + ports: + - 5432:5432 + options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 steps: - name: Checkout Repository uses: actions/checkout@v4 From 55bb53d47c4561917539377d1d7eb343112a4223 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Wed, 1 May 2024 20:27:50 +0700 Subject: [PATCH 02/20] [FIX] Fix cd workflow --- .github/workflows/cd.yml | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/.github/workflows/cd.yml b/.github/workflows/cd.yml index 47d1fb0..1153188 100644 --- a/.github/workflows/cd.yml +++ b/.github/workflows/cd.yml @@ -45,6 +45,16 @@ jobs: name: Test runs-on: ubuntu-latest needs: build + services: + postgres: + image: postgres:latest + env: + POSTGRES_USER: postgres + POSTGRES_PASSWORD: postgres + POSTGRES_DB: snackscription_review + ports: + - 5432:5432 + options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 steps: - name: Checkout Repository uses: actions/checkout@v4 From 5518184933b3197eac709d7e073426b235ec7677 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Wed, 8 May 2024 20:13:44 +0700 Subject: [PATCH 03/20] Add async function --- .../review/ReviewApplication.java | 18 ++++++++++++++++-- .../review/service/ReviewService.java | 9 +++++++++ .../service/SentimentAnalysisService.java | 8 ++++++++ .../review/service/ReviewServiceTest.java | 5 +++++ 4 files changed, 38 insertions(+), 2 deletions(-) create mode 100644 src/main/java/snackscription/review/service/SentimentAnalysisService.java diff --git a/src/main/java/snackscription/review/ReviewApplication.java b/src/main/java/snackscription/review/ReviewApplication.java index ec64e1c..4898e1d 100644 --- a/src/main/java/snackscription/review/ReviewApplication.java +++ b/src/main/java/snackscription/review/ReviewApplication.java @@ -2,14 +2,28 @@ import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; +import org.springframework.context.annotation.Bean; +import org.springframework.scheduling.annotation.EnableAsync; +import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; +import java.util.concurrent.Executor; + +@EnableAsync @SpringBootApplication public class ReviewApplication { public static void main(String[] args) { SpringApplication.run(ReviewApplication.class, args); - - } + @Bean + public Executor taskExecutor () { + ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); + executor.setCorePoolSize(1); + executor.setMaxPoolSize(1); + executor.setQueueCapacity(500); + executor.setThreadNamePrefix("GithubLookup-"); + executor.initialize(); + return executor; + } } diff --git a/src/main/java/snackscription/review/service/ReviewService.java b/src/main/java/snackscription/review/service/ReviewService.java index bcd511a..266d3ba 100644 --- a/src/main/java/snackscription/review/service/ReviewService.java +++ b/src/main/java/snackscription/review/service/ReviewService.java @@ -2,7 +2,9 @@ import java.util.List; import java.util.Optional; +import java.util.concurrent.CompletableFuture; +import org.springframework.scheduling.annotation.Async; import org.springframework.stereotype.Service; import snackscription.review.exception.InvalidStateException; @@ -14,6 +16,7 @@ @Service public class ReviewService { private ReviewRepository reviewRepository; + private SentimentAnalysisService sentimentAnalysisService; public ReviewService (ReviewRepository reviewRepository) { this.reviewRepository = reviewRepository; @@ -90,4 +93,10 @@ public Review rejectReview(String reviewId) throws Exception { review.reject(); return reviewRepository.save(review); } + + @Async + public CompletableFuture analyzeSentimentAsync(String reviewText) { + String sentiment = sentimentAnalysisService.analyze(reviewText); + return CompletableFuture.completedFuture(sentiment); + } } diff --git a/src/main/java/snackscription/review/service/SentimentAnalysisService.java b/src/main/java/snackscription/review/service/SentimentAnalysisService.java new file mode 100644 index 0000000..b9610a8 --- /dev/null +++ b/src/main/java/snackscription/review/service/SentimentAnalysisService.java @@ -0,0 +1,8 @@ +package snackscription.review.service; + +public class SentimentAnalysisService { + public String analyze(String reviewText) { + return "positive"; + } + +} diff --git a/src/test/java/snackscription/review/service/ReviewServiceTest.java b/src/test/java/snackscription/review/service/ReviewServiceTest.java index 264cb15..2187307 100644 --- a/src/test/java/snackscription/review/service/ReviewServiceTest.java +++ b/src/test/java/snackscription/review/service/ReviewServiceTest.java @@ -241,5 +241,10 @@ public void assertEqualReview(Review review1, Review review2) { assertEquals(review1.getUserId(), review2.getUserId()); assertEquals(review1.getSubscriptionBoxId(), review2.getSubscriptionBoxId()); } + + @Test + public void analyzeSentimentAsyncTest() { + + } } From 9c61cfde65a30db203e45f9ff87e9a1efb06daa2 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Wed, 8 May 2024 20:55:24 +0700 Subject: [PATCH 04/20] [REFACTOR] simplify controller API endpoint path --- .../review/controller/ReviewController.java | 20 +++++++++---------- .../controller/ReviewControllerTest.java | 16 +++++++-------- 2 files changed, 18 insertions(+), 18 deletions(-) diff --git a/src/main/java/snackscription/review/controller/ReviewController.java b/src/main/java/snackscription/review/controller/ReviewController.java index 4b89ab2..05db339 100644 --- a/src/main/java/snackscription/review/controller/ReviewController.java +++ b/src/main/java/snackscription/review/controller/ReviewController.java @@ -43,7 +43,7 @@ public ResponseEntity reviewPage() { return ResponseEntity.ok().body("Welcome to the review service!"); } - @PostMapping("/api/subscription-boxes/{subscriptionBoxId}") + @PostMapping("/subscription-boxes/{subscriptionBoxId}") public ResponseEntity createSubscriptionBoxReview(@RequestBody Map body, @PathVariable String subscriptionBoxId) { try { @@ -58,7 +58,7 @@ public ResponseEntity createSubscriptionBoxReview(@RequestBody Map> getAllPublicSubscriptionBoxReview(@PathVariable String subscriptionBoxId) { try { List reviews = reviewService.getAllSubscriptionBoxReview(subscriptionBoxId, "APPROVED"); @@ -68,7 +68,7 @@ public ResponseEntity> getAllPublicSubscriptionBoxReview(@PathVaria } } - @GetMapping("/api/subscription-boxes/{subscriptionBoxId}/users/self") + @GetMapping("/subscription-boxes/{subscriptionBoxId}/users/self") public ResponseEntity getSelfSubscriptionBoxReview(@RequestBody Map body, @PathVariable String subscriptionBoxId) { try { String userId = body.get("userId"); @@ -79,7 +79,7 @@ public ResponseEntity getSelfSubscriptionBoxReview(@RequestBody Map editSelfSubscriptionBoxId(@RequestBody Map body, @PathVariable String subscriptionBoxId) { try { String userId = body.get("userId"); @@ -93,7 +93,7 @@ public ResponseEntity editSelfSubscriptionBoxId(@RequestBody Map deleteSelfSubscriptionBoxReview(@RequestBody Map body, @PathVariable String subscriptionBoxId) { try { String userId = body.get("userId"); @@ -104,7 +104,7 @@ public ResponseEntity deleteSelfSubscriptionBoxReview(@RequestBody Map deleteSubscriptionBoxReview(@PathVariable String subscriptionBoxId, @PathVariable String userId) { try { reviewService.deleteReview(subscriptionBoxId, userId); @@ -114,17 +114,17 @@ public ResponseEntity deleteSubscriptionBoxReview(@PathVariable String s } } - @GetMapping("/api/reviews/{subsboxId}") + @GetMapping("/reviews/{subsboxId}") public List getBySubscriptionBoxId(@PathVariable String subsboxId) throws Exception { return reviewService.getAllSubscriptionBoxReview(subsboxId, null); } - @GetMapping("/api/reviews/{reviewId}") + @GetMapping("/reviews/{reviewId}") public Review getById(@PathVariable String reviewId) throws Exception { return reviewService.findById(reviewId); } - @PutMapping("/api/reviews/{reviewId}/approve") + @PutMapping("/reviews/{reviewId}/approve") public ResponseEntity approveReview(@PathVariable String reviewId) { try { Review review = reviewService.approveReview(reviewId); @@ -134,7 +134,7 @@ public ResponseEntity approveReview(@PathVariable String reviewId) { } } - @PutMapping("/api/reviews/{reviewId}/reject") + @PutMapping("/reviews/{reviewId}/reject") public ResponseEntity rejectReview(@PathVariable String reviewId) { try { Review review = reviewService.rejectReview(reviewId); diff --git a/src/test/java/snackscription/review/controller/ReviewControllerTest.java b/src/test/java/snackscription/review/controller/ReviewControllerTest.java index 764efdc..fa4260a 100644 --- a/src/test/java/snackscription/review/controller/ReviewControllerTest.java +++ b/src/test/java/snackscription/review/controller/ReviewControllerTest.java @@ -78,7 +78,7 @@ public void testCreateSubscriptionBoxReview() throws Exception{ when(reviewService.createReview(review.getRating(), review.getContent(), review.getSubscriptionBoxId(), review.getUserId())).thenReturn(review); - ResultActions result = mockMvc.perform(post("/api/subscription-boxes/{subscriptionBoxId}", review.getSubscriptionBoxId()) + ResultActions result = mockMvc.perform(post("/subscription-boxes/{subscriptionBoxId}", review.getSubscriptionBoxId()) .contentType(MediaType.APPLICATION_JSON) .content("{\"rating\": 5, \"content\": \"I love it\", \"userId\": \"user_123\"}")) .andExpect(status().isCreated()) @@ -102,7 +102,7 @@ public void testReadAllPublicSubscriptionBoxReview() throws Exception { when(reviewService.getAllSubscriptionBoxReview(subsboxId, "APPROVED")).thenReturn(approvedReviews); - String result = mockMvc.perform(get("/api/subscription-boxes/{subscriptionBoxId}", subsboxId)) + String result = mockMvc.perform(get("/subscription-boxes/{subscriptionBoxId}", subsboxId)) .andExpect(status().isOk()) .andExpect(jsonPath("$", hasSize(approvedReviews.size()))) .andReturn() @@ -149,7 +149,7 @@ public void readSelfSubscriptionBoxReview() throws Exception { when(reviewService.getReview(subsboxId, userId)).thenReturn(review); - ResultActions result = mockMvc.perform(get("/api/subscription-boxes/{subscriptionBoxId}/users/self", subsboxId) + ResultActions result = mockMvc.perform(get("/subscription-boxes/{subscriptionBoxId}/users/self", subsboxId) .contentType(MediaType.APPLICATION_JSON) .content("{\"userId\": \"user_123\"}")) .andExpect(status().isOk()) @@ -171,7 +171,7 @@ public void testEditSelfSubscriptionBoxReview() throws Exception { String newContent = "Awikwok"; when(reviewService.editReview(newRating, newContent, subsboxId, userId)).thenReturn(new Review(newRating, newContent, userId, subsboxId)); - ResultActions result = mockMvc.perform(put("/api/subscription-boxes/{subscriptionBoxId}/users/self", subsboxId) + ResultActions result = mockMvc.perform(put("/subscription-boxes/{subscriptionBoxId}/users/self", subsboxId) .contentType(MediaType.APPLICATION_JSON) .content("{\"rating\": 4, \"content\": \"Awikwok\", \"userId\": \"user_123\"}")) .andExpect(status().isOk()) @@ -191,7 +191,7 @@ public void testDeleteSelfSubscriptionBoxReview() throws Exception { doNothing().when(reviewService).deleteReview(subsboxId, userId); - ResultActions result = mockMvc.perform(delete("/api/subscription-boxes/{subscriptionBoxId}/users/self", subsboxId) + ResultActions result = mockMvc.perform(delete("/subscription-boxes/{subscriptionBoxId}/users/self", subsboxId) .contentType(MediaType.APPLICATION_JSON) .content("{\"userId\": \"user_123\"}")) .andExpect(status().isNoContent()); @@ -207,7 +207,7 @@ public void testDeleteUserSubscriptionBoxReview() throws Exception { doNothing().when(reviewService).deleteReview(subsboxId, userId); - ResultActions result = mockMvc.perform(delete("/api/subscription-boxes/{subscriptionBoxId}/users/{userId}", subsboxId, userId) + ResultActions result = mockMvc.perform(delete("/subscription-boxes/{subscriptionBoxId}/users/{userId}", subsboxId, userId) .contentType(MediaType.APPLICATION_JSON)) .andExpect(status().isNoContent()); @@ -224,7 +224,7 @@ public void testApproveReview() throws Exception { when(reviewService.approveReview(reviewId)).thenReturn(approvedReview); - ResultActions result = mockMvc.perform(put("/api/reviews/{reviewId}/approve", reviewId)) + ResultActions result = mockMvc.perform(put("/reviews/{reviewId}/approve", reviewId)) .andExpect(status().isOk()) .andExpect(jsonPath("$.rating", is(review.getRating()))) .andExpect(jsonPath("$.content", is(review.getContent()))) @@ -245,7 +245,7 @@ public void testRejectReview() throws Exception { when(reviewService.rejectReview(reviewId)).thenReturn(rejectedReview); - ResultActions result = mockMvc.perform(put("/api/reviews/{reviewId}/reject", reviewId)) + ResultActions result = mockMvc.perform(put("/reviews/{reviewId}/reject", reviewId)) .andExpect(status().isOk()) .andExpect(jsonPath("$.rating", is(review.getRating()))) .andExpect(jsonPath("$.content", is(review.getContent()))) From 124e622745d9480c2e83b90d0a9bb0976afec1ca Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Wed, 15 May 2024 15:24:08 +0700 Subject: [PATCH 05/20] [REFACTOR] Rename APO endpoint path, rename user to author, change review id attribute to composite of (subsbox, author) --- gradlew | 2 +- .../review/controller/ReviewController.java | 101 ++++---- .../snackscription/review/model/Review.java | 34 ++- .../snackscription/review/model/ReviewId.java | 25 ++ .../review/repository/ReviewRepository.java | 14 +- .../review/service/ReviewService.java | 65 +++--- src/main/resources/application-dev.properties | 2 +- .../controller/ReviewControllerTest.java | 217 ++++++++---------- .../review/model/ReviewTest.java | 6 +- .../repository/ReviewRepositoryTest.java | 36 ++- .../review/service/ReviewServiceTest.java | 140 ++++------- 11 files changed, 275 insertions(+), 367 deletions(-) create mode 100644 src/main/java/snackscription/review/model/ReviewId.java diff --git a/gradlew b/gradlew index 1aa94a4..e7c6edb 100755 --- a/gradlew +++ b/gradlew @@ -205,7 +205,7 @@ DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' # Collect all arguments for the java command: # * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments, # and any embedded shellness will be escaped. -# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be +# * For example: A author cannot expect ${Hostname} to be expanded, as it is an environment variable and will be # treated as '${Hostname}' itself on the command line. set -- \ diff --git a/src/main/java/snackscription/review/controller/ReviewController.java b/src/main/java/snackscription/review/controller/ReviewController.java index 05db339..d126faa 100644 --- a/src/main/java/snackscription/review/controller/ReviewController.java +++ b/src/main/java/snackscription/review/controller/ReviewController.java @@ -1,34 +1,17 @@ package snackscription.review.controller; -import java.util.ArrayList; import java.util.List; import java.util.Map; -import org.apache.catalina.connector.Response; -import org.springframework.beans.factory.annotation.Autowired; -import org.springframework.boot.autoconfigure.web.ServerProperties.Tomcat.Resource; import org.springframework.http.HttpStatus; import org.springframework.http.ResponseEntity; -import org.springframework.stereotype.Controller; -import org.springframework.web.bind.annotation.DeleteMapping; -import org.springframework.web.bind.annotation.GetMapping; -import org.springframework.web.bind.annotation.PathVariable; -import org.springframework.web.bind.annotation.RequestMapping; -import org.springframework.web.bind.annotation.RestController; +import org.springframework.web.bind.annotation.*; + import snackscription.review.model.Review; -import snackscription.review.repository.ReviewRepository; import snackscription.review.service.ReviewService; -import org.springframework.web.bind.annotation.RequestParam; -import org.springframework.web.bind.annotation.PostMapping; -import org.springframework.web.bind.annotation.RequestBody; -import org.springframework.web.bind.annotation.PutMapping; - - - - @RestController -@RequestMapping("/") +@RequestMapping("/reviews") public class ReviewController { @@ -43,101 +26,95 @@ public ResponseEntity reviewPage() { return ResponseEntity.ok().body("Welcome to the review service!"); } - @PostMapping("/subscription-boxes/{subscriptionBoxId}") - public ResponseEntity createSubscriptionBoxReview(@RequestBody Map body, @PathVariable String subscriptionBoxId) { - + @PostMapping("/subscription-boxes/{subsbox}") + public ResponseEntity createSubsboxReview(@RequestBody Map body, @PathVariable String subsbox) { try { - String userId = body.get("userId"); + String author = body.get("author"); int rating = Integer.parseInt(body.get("rating")); String content = body.get("content"); - Review review = reviewService.createReview(rating, content, subscriptionBoxId, userId); + Review review = reviewService.createReview(rating, content, subsbox, author); return new ResponseEntity<>(review, HttpStatus.CREATED); } catch (Exception e) { return new ResponseEntity<>(HttpStatus.BAD_REQUEST); } } - @GetMapping("/subscription-boxes/{subscriptionBoxId}") - public ResponseEntity> getAllPublicSubscriptionBoxReview(@PathVariable String subscriptionBoxId) { + @GetMapping("/subscription-boxes/{subsbox}/public") + public ResponseEntity> getPublicSubsboxReview(@PathVariable String subsbox) { try { - List reviews = reviewService.getAllSubscriptionBoxReview(subscriptionBoxId, "APPROVED"); + List reviews = reviewService.getSubsboxReview(subsbox, "APPROVED"); return new ResponseEntity<>(reviews, HttpStatus.OK); } catch (Exception e) { return new ResponseEntity<>(HttpStatus.BAD_REQUEST); } } - @GetMapping("/subscription-boxes/{subscriptionBoxId}/users/self") - public ResponseEntity getSelfSubscriptionBoxReview(@RequestBody Map body, @PathVariable String subscriptionBoxId) { + @GetMapping("/subscription-boxes/{subsbox}/users/{user}") + public ResponseEntity getSelfSubsboxReview(@RequestBody Map body, @PathVariable String subsbox, @PathVariable String user) { try { - String userId = body.get("userId"); - Review review = reviewService.getReview(subscriptionBoxId, userId); + String sender = body.get("author"); // TODO: nanti pakai JWT token untuk ambil sendernya + if (!authenticate(sender, user)) { + return new ResponseEntity<>(HttpStatus.UNAUTHORIZED); + } + Review review = reviewService.getReview(subsbox, user); return new ResponseEntity<>(review, HttpStatus.OK); } catch (Exception e) { return new ResponseEntity<>(HttpStatus.BAD_REQUEST); } } - @PutMapping("/subscription-boxes/{subscriptionBoxId}/users/self") - public ResponseEntity editSelfSubscriptionBoxId(@RequestBody Map body, @PathVariable String subscriptionBoxId) { + @PutMapping("/subscription-boxes/{subsbox}/users/{user}") + public ResponseEntity editReview(@RequestBody Map body, @PathVariable String subsbox, @PathVariable String user) { try { - String userId = body.get("userId"); + String sender = body.get("author"); // TODO: nanti pakai JWT token untuk ambil sendernya + if (!authenticate(sender, user)) { + return new ResponseEntity<>(HttpStatus.UNAUTHORIZED); + } + int rating = Integer.parseInt(body.get("rating")); String content = body.get("content"); - Review review = reviewService.editReview(rating, content, subscriptionBoxId, userId); + Review review = reviewService.editReview(rating, content, subsbox, user); return new ResponseEntity<>(review, HttpStatus.OK); } catch (Exception e) { return new ResponseEntity<>(HttpStatus.BAD_REQUEST); } } - @DeleteMapping("/subscription-boxes/{subscriptionBoxId}/users/self") - public ResponseEntity deleteSelfSubscriptionBoxReview(@RequestBody Map body, @PathVariable String subscriptionBoxId) { - try { - String userId = body.get("userId"); - reviewService.deleteReview(subscriptionBoxId, userId); - return new ResponseEntity<>(HttpStatus.NO_CONTENT); - } catch (Exception e) { - return new ResponseEntity<>(HttpStatus.BAD_REQUEST); - } + private boolean authenticate(String sender, String user) { + return true; } - @DeleteMapping("/subscription-boxes/{subscriptionBoxId}/users/{userId}") - public ResponseEntity deleteSubscriptionBoxReview(@PathVariable String subscriptionBoxId, @PathVariable String userId) { + @DeleteMapping("/subscription-boxes/{subsbox}/users/{user}") + public ResponseEntity deleteReview(@PathVariable String subsbox, @PathVariable String user) { try { - reviewService.deleteReview(subscriptionBoxId, userId); + reviewService.deleteReview(subsbox, user); return new ResponseEntity<>(HttpStatus.NO_CONTENT); } catch (Exception e) { return new ResponseEntity<>(HttpStatus.BAD_REQUEST); } } - @GetMapping("/reviews/{subsboxId}") - public List getBySubscriptionBoxId(@PathVariable String subsboxId) throws Exception { - return reviewService.getAllSubscriptionBoxReview(subsboxId, null); - } - - @GetMapping("/reviews/{reviewId}") - public Review getById(@PathVariable String reviewId) throws Exception { - return reviewService.findById(reviewId); + @GetMapping("/subscription-boxes/{subsbox}") + public List getSubsboxReview(@PathVariable String subsbox) throws Exception { + return reviewService.getSubsboxReview(subsbox, null); } - @PutMapping("/reviews/{reviewId}/approve") - public ResponseEntity approveReview(@PathVariable String reviewId) { + @PutMapping("/subscription-boxes/{subsbox}/users/{user}/approve") + public ResponseEntity approveReview(@PathVariable String subsbox, @PathVariable String user) { try { - Review review = reviewService.approveReview(reviewId); + Review review = reviewService.approveReview(subsbox, user); return new ResponseEntity<>(review, HttpStatus.OK); } catch (Exception e) { return new ResponseEntity<>(HttpStatus.BAD_REQUEST); } } - @PutMapping("/reviews/{reviewId}/reject") - public ResponseEntity rejectReview(@PathVariable String reviewId) { + @PutMapping("/subscription-boxes/{subsbox}/users/{user}/reject") + public ResponseEntity rejectReview(@PathVariable String subsbox, @PathVariable String user) { try { - Review review = reviewService.rejectReview(reviewId); + Review review = reviewService.rejectReview(subsbox, user); return new ResponseEntity<>(review, HttpStatus.OK); } catch (Exception e) { return new ResponseEntity<>(HttpStatus.BAD_REQUEST); diff --git a/src/main/java/snackscription/review/model/Review.java b/src/main/java/snackscription/review/model/Review.java index 572d7a5..0ef1c69 100644 --- a/src/main/java/snackscription/review/model/Review.java +++ b/src/main/java/snackscription/review/model/Review.java @@ -1,19 +1,15 @@ package snackscription.review.model; -import lombok.Getter; -import lombok.Setter; - -import java.util.UUID; +import lombok.Data; import jakarta.persistence.*; -@Getter -@Setter +@Data @Entity -@Table(name = "review") +@Table public class Review { - @Id - private String id; + @EmbeddedId + private ReviewId id; @Column(name = "rating", nullable = false) private int rating; @@ -24,22 +20,14 @@ public class Review { @Column(name = "state", nullable = false) private ReviewState state; - @Column(name="user_id", nullable = false) - private String userId; - - @Column(name="subsbox_id", nullable = false) - private String subscriptionBoxId; - public Review() { } - public Review(int rating, String content, String userId, String subscriptionBoxId) { - this.id = UUID.randomUUID().toString(); + public Review(int rating, String content, String subsbox, String user) { + this.id = new ReviewId(subsbox, user); this.rating = rating; this.content = content; this.state = ReviewState.PENDING; - this.userId = userId; - this.subscriptionBoxId = subscriptionBoxId; } public void editReview(int rating, String content) { @@ -61,4 +49,12 @@ public void approve() { public void reject() { this.state.reject(this); } + + public String getSubsbox() { + return this.id.getSubsbox(); + } + + public String getAuthor() { + return this.id.getAuthor(); + } } \ No newline at end of file diff --git a/src/main/java/snackscription/review/model/ReviewId.java b/src/main/java/snackscription/review/model/ReviewId.java new file mode 100644 index 0000000..d1656fb --- /dev/null +++ b/src/main/java/snackscription/review/model/ReviewId.java @@ -0,0 +1,25 @@ +package snackscription.review.model; + +import jakarta.persistence.Column; +import jakarta.persistence.Embeddable; +import lombok.Data; + +import java.io.Serializable; + +@Data +@Embeddable +public class ReviewId implements Serializable { + @Column(name = "subsbox", nullable = false) + private String subsbox; + + @Column(name = "author", nullable = false) + private String author; + + public ReviewId(String subsbox, String author) { + this.subsbox = subsbox; + this.author = author; + } + public ReviewId() { + + } +} diff --git a/src/main/java/snackscription/review/repository/ReviewRepository.java b/src/main/java/snackscription/review/repository/ReviewRepository.java index bb448bf..096b317 100644 --- a/src/main/java/snackscription/review/repository/ReviewRepository.java +++ b/src/main/java/snackscription/review/repository/ReviewRepository.java @@ -2,14 +2,14 @@ import java.util.List; import org.springframework.data.jpa.repository.JpaRepository; -import org.springframework.data.jpa.repository.Query; -import org.springframework.data.repository.CrudRepository; import snackscription.review.model.Review; +import snackscription.review.model.ReviewId; import snackscription.review.model.ReviewState; -public interface ReviewRepository extends JpaRepository { - List findBySubscriptionBoxId(String subsboxId); - List findBySubscriptionBoxIdAndState(String subsboxId, ReviewState state); - Review findBySubscriptionBoxIdAndUserId(String subsboxId, String userId); - void deleteBySubscriptionBoxIdAndUserId(String subsboxId, String userId); +public interface ReviewRepository extends JpaRepository { + List findByIdSubsbox(String subsbox); + List findByIdAuthor(String author); + List findByIdSubsboxAndState(String subsbox, ReviewState state); + Review findByIdSubsboxAndIdAuthor(String subsbox, String author); + void deleteByIdSubsboxAndIdAuthor(String subsbox, String author); } \ No newline at end of file diff --git a/src/main/java/snackscription/review/service/ReviewService.java b/src/main/java/snackscription/review/service/ReviewService.java index bcd511a..64462a3 100644 --- a/src/main/java/snackscription/review/service/ReviewService.java +++ b/src/main/java/snackscription/review/service/ReviewService.java @@ -8,6 +8,7 @@ import snackscription.review.exception.InvalidStateException; import snackscription.review.exception.ReviewNotFoundException; import snackscription.review.model.Review; +import snackscription.review.model.ReviewId; import snackscription.review.model.ReviewState; import snackscription.review.repository.ReviewRepository; @@ -19,45 +20,35 @@ public ReviewService (ReviewRepository reviewRepository) { this.reviewRepository = reviewRepository; } - public Review findById(String reviewId) throws ReviewNotFoundException { - Optional oReview = reviewRepository.findById(reviewId); - - if (oReview.isEmpty()) { - throw new ReviewNotFoundException(); - } - - return oReview.get(); - } - - public List findBySubscriptionBoxId(String subscriptionBoxId) { - return reviewRepository.findBySubscriptionBoxId(subscriptionBoxId); - } - public Review createReview(int rating, String content, String subscriptionBoxId, String userId) throws Exception { - Review review = new Review(rating, content, userId, subscriptionBoxId); + Review review = new Review(rating, content, subscriptionBoxId, userId); reviewRepository.save(review); return review; } - public List getAllSubscriptionBoxReview(String subscriptionBoxId, String state) throws Exception { + public Review getReview(String subsbox, String user) throws Exception { + Optional oreview = reviewRepository.findById(new ReviewId(user, subsbox)); + if (oreview.isEmpty()) { + throw new ReviewNotFoundException(); + } + return oreview.get(); + } + + public List getSubsboxReview(String subscriptionBoxId, String state) throws Exception { if (state == null) { - return reviewRepository.findBySubscriptionBoxId(subscriptionBoxId); + return reviewRepository.findByIdSubsbox(subscriptionBoxId); } else { state = state.toUpperCase(); ReviewState reviewState = Enum.valueOf(ReviewState.class, state); if (reviewState == null) { throw new InvalidStateException(); } - return reviewRepository.findBySubscriptionBoxIdAndState(subscriptionBoxId, reviewState); + return reviewRepository.findByIdSubsboxAndState(subscriptionBoxId, reviewState); } } - public Review getReview(String subscriptionBoxId, String userId) throws Exception { - return reviewRepository.findBySubscriptionBoxIdAndUserId(subscriptionBoxId, userId); - } - public Review editReview(int rating, String content, String subscriptionBoxId, String userId) throws Exception { - Review review = reviewRepository.findBySubscriptionBoxIdAndUserId(subscriptionBoxId, userId); + Review review = reviewRepository.findByIdSubsboxAndIdAuthor(subscriptionBoxId, userId); if (review == null) { throw new ReviewNotFoundException(); @@ -69,25 +60,25 @@ public Review editReview(int rating, String content, String subscriptionBoxId, S return reviewRepository.save(review); } - public void deleteReview(String subscriptionBoxId, String userId) throws Exception { - Review review = reviewRepository.findBySubscriptionBoxIdAndUserId(subscriptionBoxId, userId); - - if (review == null) { - throw new ReviewNotFoundException(); - } - - reviewRepository.delete(review); - } - - public Review approveReview(String reviewId) throws Exception { - Review review = findById(reviewId); + public Review approveReview(String subsbox, String user) throws Exception { + Review review = getReview(subsbox, user); review.approve(); return reviewRepository.save(review); } - public Review rejectReview(String reviewId) throws Exception { - Review review = findById(reviewId); + public Review rejectReview(String subsbox, String user) throws Exception { + Review review = getReview(subsbox, user); review.reject(); return reviewRepository.save(review); } + + public void deleteReview(String subsbox, String user) throws Exception { + Review review = reviewRepository.findByIdSubsboxAndIdAuthor(subsbox, user); + + if (review == null) { + throw new ReviewNotFoundException(); + } + + reviewRepository.delete(review); + } } diff --git a/src/main/resources/application-dev.properties b/src/main/resources/application-dev.properties index dfdbb9f..083d035 100644 --- a/src/main/resources/application-dev.properties +++ b/src/main/resources/application-dev.properties @@ -1,4 +1,4 @@ -spring.datasource.url=jdbc:postgresql://localhost:5432/snackscription_review +spring.datasource.url=jdbc:postgresql://localhost:5433/snackscription_review spring.datasource.username=postgres spring.datasource.password=postgres spring.jpa.hibernate.ddl-auto=create-drop diff --git a/src/test/java/snackscription/review/controller/ReviewControllerTest.java b/src/test/java/snackscription/review/controller/ReviewControllerTest.java index fa4260a..bf0ccca 100644 --- a/src/test/java/snackscription/review/controller/ReviewControllerTest.java +++ b/src/test/java/snackscription/review/controller/ReviewControllerTest.java @@ -4,7 +4,6 @@ import static org.mockito.Mockito.verify; import java.util.ArrayList; -import java.util.Collections; import java.util.Comparator; import java.util.List; @@ -12,29 +11,21 @@ import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; -import org.mockito.internal.stubbing.answers.DoesNothing; 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.http.MediaType; import org.springframework.test.context.junit.jupiter.SpringExtension; import org.springframework.test.web.servlet.MockMvc; -import org.springframework.test.web.servlet.MvcResult; import org.springframework.test.web.servlet.ResultActions; -import static org.junit.jupiter.api.Assertions.*; -import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.status; -import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.content; -import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.jsonPath; -import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.delete; -import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.get; -import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.post; -import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.put; +import static org.junit.jupiter.api.Assertions.*; +import static org.springframework.test.web.servlet.result.MockMvcResultMatchers.*; +import static org.springframework.test.web.servlet.request.MockMvcRequestBuilders.*; import static org.hamcrest.Matchers.hasSize; import static org.hamcrest.Matchers.is; import static org.mockito.Mockito.when; -import org.springframework.test.web.servlet.result.JsonPathResultMatchers; import snackscription.review.model.Review; import snackscription.review.model.ReviewState; import snackscription.review.service.ReviewService; @@ -54,11 +45,11 @@ public class ReviewControllerTest { public void setUp() { this.reviews = new ArrayList<>(); - Review review1 = new Review(5, "I love it", "user_123", "subsbox_123"); - Review review2 = new Review(1, "I hate it", "user_124", "subsbox_123"); - Review review3 = new Review(2, "Hmmmm idk", "user_124", "subsbox_124"); - Review review4 = new Review(3, "It's okay", "user_125", "subsbox_124"); - Review review5 = new Review(4, "I like it", "user_126", "subsbox_124"); + Review review1 = new Review(5, "I love it", "subsbox_123", "user_123"); + Review review2 = new Review(1, "I hate it", "subsbox_123", "user_124"); + Review review3 = new Review(2, "Hmmmm idk", "subsbox_124", "user_124"); + Review review4 = new Review(3, "It's okay", "subsbox_124", "user_125"); + Review review5 = new Review(4, "I like it", "subsbox_124", "user_126"); review1.setState(ReviewState.PENDING); review4.setState(ReviewState.APPROVED); @@ -73,36 +64,36 @@ public void setUp() { } @Test - public void testCreateSubscriptionBoxReview() throws Exception{ + public void testCreateSubsboxReview() throws Exception{ Review review = reviews.getFirst(); - when(reviewService.createReview(review.getRating(), review.getContent(), review.getSubscriptionBoxId(), review.getUserId())).thenReturn(review); + when(reviewService.createReview(review.getRating(), review.getContent(), review.getId().getSubsbox(), review.getId().getAuthor())).thenReturn(review); - ResultActions result = mockMvc.perform(post("/subscription-boxes/{subscriptionBoxId}", review.getSubscriptionBoxId()) + ResultActions result = mockMvc.perform(post("/reviews/subscription-boxes/{subsbox}", review.getSubsbox()) .contentType(MediaType.APPLICATION_JSON) - .content("{\"rating\": 5, \"content\": \"I love it\", \"userId\": \"user_123\"}")) + .content("{\"rating\": 5, \"content\": \"I love it\", \"author\": \"user_123\"}")) .andExpect(status().isCreated()) .andExpect(jsonPath("$.rating", is(5))) .andExpect(jsonPath("$.content", is("I love it"))) - .andExpect(jsonPath("$.userId", is("user_123"))) - .andExpect(jsonPath("$.subscriptionBoxId", is("subsbox_123"))); + .andExpect(jsonPath("$.author", is("user_123"))) + .andExpect(jsonPath("$.subsbox", is("subsbox_123"))); - verify(reviewService).createReview(review.getRating(), review.getContent(), review.getSubscriptionBoxId(), review.getUserId()); + verify(reviewService).createReview(review.getRating(), review.getContent(), review.getSubsbox(), review.getAuthor()); } @Test public void testReadAllPublicSubscriptionBoxReview() throws Exception { List approvedReviews = new ArrayList<>(); - String subsboxId = "subsbox_124"; + String subsbox = "subsbox_124"; for (Review review : reviews) { - if (review.getSubscriptionBoxId().equals(subsboxId) && review.getState().equals(ReviewState.APPROVED)) { + if (review.getSubsbox().equals(subsbox) && review.getState().equals(ReviewState.APPROVED)) { approvedReviews.add(review); } } - when(reviewService.getAllSubscriptionBoxReview(subsboxId, "APPROVED")).thenReturn(approvedReviews); + when(reviewService.getSubsboxReview(subsbox, "APPROVED")).thenReturn(approvedReviews); - String result = mockMvc.perform(get("/subscription-boxes/{subscriptionBoxId}", subsboxId)) + String result = mockMvc.perform(get("/reviews/subscription-boxes/{subsbox}/public", subsbox)) .andExpect(status().isOk()) .andExpect(jsonPath("$", hasSize(approvedReviews.size()))) .andReturn() @@ -114,146 +105,138 @@ public void testReadAllPublicSubscriptionBoxReview() throws Exception { String prefixMatcher = String.format("$[%d]", i); int rating = JsonPath.read(result, prefixMatcher + ".rating"); String content = JsonPath.read(result, prefixMatcher + ".content"); - String userId = JsonPath.read(result, prefixMatcher + ".userId"); - String curSubscriptionBoxId = JsonPath.read(result, prefixMatcher + ".subscriptionBoxId"); + String author = JsonPath.read(result, prefixMatcher + ".author"); + String curSubscriptionBoxId = JsonPath.read(result, prefixMatcher + ".subsbox"); - Review review = new Review(rating, content, userId, curSubscriptionBoxId); + Review review = new Review(rating, content, curSubscriptionBoxId, author); foundReviews.add(review); } - Comparator cmp = new Comparator() { - @Override - public int compare(Review o1, Review o2) { - return o1.getUserId().compareTo(o2.getUserId()); - } - }; - + Comparator cmp = Comparator.comparing(Review::getAuthor); approvedReviews.sort(cmp); foundReviews.sort(cmp); for (int i=0; i curReviews = new ArrayList<>(); -// String subscriptionBoxId = this.reviews.getFirst().getSubscriptionBoxId(); +// String subscriptionBoxId = this.reviews.getFirst().getSubsbox(); // for (Review review : this.reviews) { -// if (review.getSubscriptionBoxId().equals(subscriptionBoxId)) { +// if (review.getSubsbox().equals(subscriptionBoxId)) { // curReviews.add(review); // } // } @@ -335,7 +318,7 @@ public void testRejectReview() throws Exception { // Comparator cmp = new Comparator() { // @Override // public int compare(Review o1, Review o2) { -// return o1.getUserId().compareTo(o2.getUserId()); +// return o1.getAuthor().compareTo(o2.getAuthor()); // } // }; @@ -345,8 +328,8 @@ public void testRejectReview() throws Exception { // for (int i=0; i(); - - Review review1 = new Review(5, "I love it", "user_123", "subsbox_123"); - Review review2 = new Review(1, "I hate it", "user_124", "subsbox_123"); - Review review3 = new Review(2, "Hmmmm idk", "user_124", "subsbox_124"); - Review review4 = new Review(3, "It's okay", "user_125", "subsbox_124"); - Review review5 = new Review(4, "I like it", "user_126", "subsbox_124"); + + Review review1 = new Review(5, "I love it", "subsbox_123", "user_123"); + Review review2 = new Review(1, "I hate it", "subsbox_123", "user_124"); + Review review3 = new Review(2, "Hmmmm idk", "subsbox_124", "user_124"); + Review review4 = new Review(3, "It's okay", "subsbox_124", "user_125"); + Review review5 = new Review(4, "I like it", "subsbox_124", "user_126"); review1.setState(ReviewState.PENDING); review4.setState(ReviewState.APPROVED); @@ -50,14 +49,14 @@ public void setUp() { public void testFindBySubscriptionBoxId() { List curReviews = new ArrayList<>(); - String subsbox_id = this.reviews.getFirst().getSubscriptionBoxId(); + String subsbox_id = this.reviews.getFirst().getSubsbox(); for (Review review : this.reviews) { - if (review.getSubscriptionBoxId().equals(subsbox_id)) { + if (review.getSubsbox().equals(subsbox_id)) { curReviews.add(review); } } - List foundReviews = reviewRepository.findBySubscriptionBoxId(subsbox_id); + List foundReviews = reviewRepository.findByIdSubsbox(subsbox_id); assertEquals(curReviews.size(), foundReviews.size()); for (int i=0; i curReviews = new ArrayList<>(); - String subsbox_id = this.reviews.getFirst().getSubscriptionBoxId(); + String subsbox_id = this.reviews.getFirst().getSubsbox(); for (Review review : this.reviews) { - if (review.getSubscriptionBoxId().equals(subsbox_id) && review.getState().equals(ReviewState.APPROVED)){ + if (review.getSubsbox().equals(subsbox_id) && review.getState().equals(ReviewState.APPROVED)){ curReviews.add(review); } } - List foundReviews = reviewRepository.findBySubscriptionBoxIdAndState(subsbox_id, ReviewState.APPROVED); + List foundReviews = reviewRepository.findByIdSubsboxAndState(subsbox_id, ReviewState.APPROVED); assertEquals(curReviews.size(), foundReviews.size()); for (int i=0; i reviews; - // @Test - // public void testGetAllSubscriptionBoxReview() { - // ReviewService reviewService = new ReviewService(reviewRepo); - - // Optional review = Optional.of(new Review( - // 5, "amazing", "user1", "subsboxId" - // )); - - // when(reviewRepo.findById("subsboxId")).thenReturn(review); - - // Review foundReview = reviewService.getAllSubscriptionBoxReview("subsboxId"); - - // assertEquals(review.get(), foundReview); - - // verify(reviewRepo).findBySubscriptionBoxId("subsboxId"); - - // } - @BeforeEach public void setUp() { reviewService = new ReviewService(reviewRepo); - Review review1 = new Review(5, "I love it", "user_123", "subsbox_123"); - Review review2 = new Review(1, "I hate it", "user_124", "subsbox_123"); - Review review3 = new Review(2, "Hmmmm idk", "user_124", "subsbox_124"); - Review review4 = new Review(3, "It's okay", "user_125", "subsbox_124"); - Review review5 = new Review(4, "I like it", "user_126", "subsbox_124"); + Review review1 = new Review(5, "I love it", "subsbox_123", "user_123"); + Review review2 = new Review(1, "I hate it", "subsbox_123", "user_124"); + Review review3 = new Review(2, "Hmmmm idk", "subsbox_124", "user_124"); + Review review4 = new Review(3, "It's okay", "subsbox_124", "user_125"); + Review review5 = new Review(4, "I like it", "subsbox_124", "user_126"); review1.setState(ReviewState.PENDING); review4.setState(ReviewState.APPROVED); @@ -78,61 +54,25 @@ public void setUp() { } @Test - public void getReviewById() throws Exception { - - - Optional review = Optional.of(new Review( - 5, "amazing", "user1", "subsboxId" - )); - - String reviewId = review.get().getId(); - - when(reviewRepo.findById(reviewId)).thenReturn(review); - - Review foundReview = reviewService.findById(reviewId); - - assertEquals(foundReview, review.get()); - - verify(reviewRepo).findById(reviewId); - } - - @Test - public void getReviewByIdNotFound() { - ReviewService reviewService = new ReviewService(reviewRepo); - - Optional review = Optional.empty(); - - String reviewId = "reviewId"; - - when(reviewRepo.findById(reviewId)).thenReturn(review); - - assertThrows(ReviewNotFoundException.class, () -> { - reviewService.findById(reviewId); - }); - - verify(reviewRepo).findById(reviewId); - } - - @Test - public void getReviewsBySubscriptionBoxId() { + public void getReviewsBySubscriptionBoxId() throws Exception { ReviewService reviewService = new ReviewService(reviewRepo); List curReviews = new ArrayList<>(); - String subscriptionBoxId = this.reviews.getFirst().getSubscriptionBoxId(); + String subscriptionBoxId = this.reviews.getFirst().getSubsbox(); for (Review review : this.reviews) { - if (review.getSubscriptionBoxId().equals(subscriptionBoxId)) { + if (review.getSubsbox().equals(subscriptionBoxId)) { curReviews.add(review); } } - when(reviewRepo.findBySubscriptionBoxId(subscriptionBoxId)).thenReturn(curReviews); + when(reviewRepo.findByIdSubsbox(subscriptionBoxId)).thenReturn(curReviews); - List foundReviews = reviewService.findBySubscriptionBoxId(subscriptionBoxId); + List foundReviews = reviewService.getSubsboxReview(subscriptionBoxId, null); assertEquals(curReviews, foundReviews); - verify(reviewRepo).findBySubscriptionBoxId(subscriptionBoxId); + verify(reviewRepo).findByIdSubsbox(subscriptionBoxId); } @Test @@ -144,8 +84,8 @@ public void testCreateReview() throws Exception { Review savedReview = reviewService.createReview( review.getRating(), review.getContent(), - review.getSubscriptionBoxId(), - review.getUserId()); + review.getSubsbox(), + review.getAuthor()); assertEqualReview(review, savedReview); @@ -153,84 +93,82 @@ public void testCreateReview() throws Exception { } @Test - public void testGetAllSubscriptionBoxReview() throws Exception { - String subscriptionBoxId = this.reviews.getFirst().getSubscriptionBoxId(); + public void testgetSubsboxReview() throws Exception { + String subscriptionBoxId = this.reviews.getFirst().getSubsbox(); List curReviews = new ArrayList<>(); for (Review review : this.reviews) { - if (review.getSubscriptionBoxId().equals(subscriptionBoxId)) { + if (review.getSubsbox().equals(subscriptionBoxId)) { curReviews.add(review); } } - when(reviewRepo.findBySubscriptionBoxId(subscriptionBoxId)).thenReturn(curReviews); + when(reviewRepo.findByIdSubsbox(subscriptionBoxId)).thenReturn(curReviews); - List foundReviews = reviewService.getAllSubscriptionBoxReview(subscriptionBoxId, null); + List foundReviews = reviewService.getSubsboxReview(subscriptionBoxId, null); assertEquals(curReviews, foundReviews); - verify(reviewRepo).findBySubscriptionBoxId(subscriptionBoxId); + verify(reviewRepo).findByIdSubsbox(subscriptionBoxId); } @Test - public void testGetAllSubscriptionBoxReviewApproved() throws Exception { - String subscriptionBoxId = this.reviews.getFirst().getSubscriptionBoxId(); + public void testgetSubsboxReviewApproved() throws Exception { + String subscriptionBoxId = this.reviews.getFirst().getSubsbox(); List cuReviews = new ArrayList<>(); for (Review review : this.reviews) { - if (review.getSubscriptionBoxId().equals(subscriptionBoxId) && review.getState().equals(ReviewState.APPROVED)) { + if (review.getSubsbox().equals(subscriptionBoxId) && review.getState().equals(ReviewState.APPROVED)) { cuReviews.add(review); } } - when(reviewRepo.findBySubscriptionBoxIdAndState(subscriptionBoxId, ReviewState.APPROVED)).thenReturn(cuReviews); + when(reviewRepo.findByIdSubsboxAndState(subscriptionBoxId, ReviewState.APPROVED)).thenReturn(cuReviews); - List foundReviews = reviewService.getAllSubscriptionBoxReview(subscriptionBoxId, "APPROVED"); + List foundReviews = reviewService.getSubsboxReview(subscriptionBoxId, "APPROVED"); assertEquals(cuReviews, foundReviews); - verify(reviewRepo).findBySubscriptionBoxIdAndState(subscriptionBoxId, ReviewState.APPROVED); + verify(reviewRepo).findByIdSubsboxAndState(subscriptionBoxId, ReviewState.APPROVED); } @Test public void testEditReview() throws Exception { Review review = reviews.getFirst(); - String subscriptionBoxId = review.getSubscriptionBoxId(); - String userId = review.getUserId(); + String subsbox = review.getSubsbox(); + String author = review.getAuthor(); int newRating = 1; String newContent = "Changed content"; - Review newReview = new Review(newRating, newContent, userId, subscriptionBoxId); + Review newReview = new Review(newRating, newContent, author, subsbox); newReview.setId(review.getId()); - when(reviewRepo.findBySubscriptionBoxIdAndUserId(subscriptionBoxId, userId)).thenReturn(review); + when(reviewRepo.findByIdSubsboxAndIdAuthor(subsbox, author)).thenReturn(review); when(reviewRepo.save(any(Review.class))).thenReturn(newReview); - Review editedReview = reviewService.editReview(newRating, newContent, subscriptionBoxId, userId); + Review editedReview = reviewService.editReview(newRating, newContent, subsbox, author); assertEquals(newRating, editedReview.getRating()); assertEquals(newContent, editedReview.getContent()); - assertEquals(subscriptionBoxId, editedReview.getSubscriptionBoxId()); - assertEquals(userId, editedReview.getUserId()); + assertEquals(subsbox, editedReview.getSubsbox()); + assertEquals(author, editedReview.getAuthor()); assertEquals(review.getId(), editedReview.getId()); } @Test public void testDeleteReview() throws Exception { - String subscriptionBoxId = this.reviews.getFirst().getSubscriptionBoxId(); - String userId = this.reviews.getFirst().getUserId(); + String subsbox = this.reviews.getFirst().getSubsbox(); + String author = this.reviews.getFirst().getAuthor(); Review review = reviews.getFirst(); - when(reviewRepo.findBySubscriptionBoxIdAndUserId(subscriptionBoxId, userId)).thenReturn(review); - - reviewService.deleteReview(subscriptionBoxId, userId); + when(reviewRepo.findByIdSubsboxAndIdAuthor(subsbox, author)).thenReturn(review); - when(reviewRepo.findBySubscriptionBoxIdAndUserId(subscriptionBoxId, userId)).thenReturn(null); + reviewService.deleteReview(subsbox, author); - assertNull(reviewService.getReview(subscriptionBoxId, userId)); + assertThrows(ReviewNotFoundException.class, () -> reviewService.getReview(subsbox, author)); verify(reviewRepo).delete(review); } @@ -238,8 +176,8 @@ public void testDeleteReview() throws Exception { public void assertEqualReview(Review review1, Review review2) { assertEquals(review1.getRating(), review2.getRating()); assertEquals(review1.getContent(), review2.getContent()); - assertEquals(review1.getUserId(), review2.getUserId()); - assertEquals(review1.getSubscriptionBoxId(), review2.getSubscriptionBoxId()); + assertEquals(review1.getAuthor(), review2.getAuthor()); + assertEquals(review1.getSubsbox(), review2.getSubsbox()); } } From a416bbc9e5a33fdae3baf4bf4358078c4e139186 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Wed, 15 May 2024 15:35:17 +0700 Subject: [PATCH 06/20] [FIX] Change application properties --- src/main/resources/application-dev.properties | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/resources/application-dev.properties b/src/main/resources/application-dev.properties index 083d035..dfdbb9f 100644 --- a/src/main/resources/application-dev.properties +++ b/src/main/resources/application-dev.properties @@ -1,4 +1,4 @@ -spring.datasource.url=jdbc:postgresql://localhost:5433/snackscription_review +spring.datasource.url=jdbc:postgresql://localhost:5432/snackscription_review spring.datasource.username=postgres spring.datasource.password=postgres spring.jpa.hibernate.ddl-auto=create-drop From 032d5df9c612c72fc7be2713c6c143c5fb51164f Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Wed, 15 May 2024 15:57:21 +0700 Subject: [PATCH 07/20] [FIX] Fix ci-cd application properties --- .github/workflows/cd.yml | 2 ++ .github/workflows/ci.yml | 6 +++++- src/main/resources/application-dev.properties | 2 +- src/main/resources/application.properties | 4 +++- 4 files changed, 11 insertions(+), 3 deletions(-) diff --git a/.github/workflows/cd.yml b/.github/workflows/cd.yml index ebdb71d..30e9471 100644 --- a/.github/workflows/cd.yml +++ b/.github/workflows/cd.yml @@ -70,6 +70,8 @@ jobs: ./gradlew check --info --stacktrace ./gradlew test ./gradlew jacocoTestReport + env: + PRODUCTION: test # (Optional) Add steps for generating coverage report and other post-test tasks publish: diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 6dbb8fa..de84db9 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -70,8 +70,10 @@ jobs: distribution: "temurin" java-version: "21" cache: "gradle" + - name: Make gradlew executable run: chmod +x ./gradlew + - name: Cache Gradle dependencies uses: actions/cache@v4 with: @@ -85,4 +87,6 @@ jobs: ./gradlew check --info --stacktrace ./gradlew test ./gradlew jacocoTestReport - # (Optional) Add steps for generating coverage report and other post-test tasks + env: + PRODUCTION: test + # (Optional) Add steps for generating coverage report and other post-test tasks diff --git a/src/main/resources/application-dev.properties b/src/main/resources/application-dev.properties index dfdbb9f..083d035 100644 --- a/src/main/resources/application-dev.properties +++ b/src/main/resources/application-dev.properties @@ -1,4 +1,4 @@ -spring.datasource.url=jdbc:postgresql://localhost:5432/snackscription_review +spring.datasource.url=jdbc:postgresql://localhost:5433/snackscription_review spring.datasource.username=postgres spring.datasource.password=postgres spring.jpa.hibernate.ddl-auto=create-drop diff --git a/src/main/resources/application.properties b/src/main/resources/application.properties index 338aaa3..1bd6102 100644 --- a/src/main/resources/application.properties +++ b/src/main/resources/application.properties @@ -1,2 +1,4 @@ spring.application.name=review -spring.profiles.active=${PRODUCTION:dev} \ No newline at end of file +spring.profiles.active=${PRODUCTION:dev} +management.endpoint.info.enabled=true +management.endpoints.web.exposure.include=health,metrics,prometheus,loggers \ No newline at end of file From f45549cc5f2c121f0aaf998cbe0b73d8e6c0fca2 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Wed, 15 May 2024 20:53:00 +0700 Subject: [PATCH 08/20] Add prometheus & grafana config --- Dockerfile | 3 +- build.gradle.kts | 2 + deployment.yaml | 22 +++++++++++ grafana-deployment.yaml | 30 +++++++++++++++ prometheus-development.yaml | 38 +++++++++++++++++++ prometheus.yml | 8 ++++ service.yaml | 11 ++++++ .../review/ReviewAppConfig.java | 14 +++++++ .../review/service/ReviewService.java | 5 +++ 9 files changed, 131 insertions(+), 2 deletions(-) create mode 100644 deployment.yaml create mode 100644 grafana-deployment.yaml create mode 100644 prometheus-development.yaml create mode 100644 prometheus.yml create mode 100644 service.yaml create mode 100644 src/main/java/snackscription/review/ReviewAppConfig.java diff --git a/Dockerfile b/Dockerfile index db9b952..516b0e1 100644 --- a/Dockerfile +++ b/Dockerfile @@ -10,7 +10,6 @@ ENV JDBC_DATABASE_URL ${JDBC_DATABASE_URL} ENV JDBC_DATABASE_USERNAME ${JDBC_DATABASE_USERNAME} WORKDIR /app -COPY ./review-0.0.1-SNAPSHOT.jar /app -RUN ls -la +COPY build/libs/review-0.0.1-SNAPSHOT.jar /app/review-0.0.1-SNAPSHOT.jar EXPOSE 8080 CMD ["java","-jar","review-0.0.1-SNAPSHOT.jar"] \ No newline at end of file diff --git a/build.gradle.kts b/build.gradle.kts index c18a996..23ef6a4 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -26,6 +26,8 @@ dependencies { implementation("org.springframework.boot:spring-boot-starter-data-jpa") implementation("org.springframework.boot:spring-boot-starter-thymeleaf") implementation("org.springframework.boot:spring-boot-starter-web") + implementation("org.springframework.boot:spring-boot-starter-actuator") + implementation("io.micrometer:micrometer-registry-prometheus") compileOnly("org.projectlombok:lombok") developmentOnly("org.springframework.boot:spring-boot-devtools") runtimeOnly("org.postgresql:postgresql") diff --git a/deployment.yaml b/deployment.yaml new file mode 100644 index 0000000..a4401a5 --- /dev/null +++ b/deployment.yaml @@ -0,0 +1,22 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: snackscription-review-deployment +spec: + replicas: 3 + selector: + matchLabels: + app: snackscription-review + template: + metadata: + labels: + app: snackscription-review + spec: + containers: + - name: snackscription-review + image: asteriskzie/snackscription-review:latest + ports: + - containerPort: 8080 + env: + - name: PRODUCTION + value: prod \ No newline at end of file diff --git a/grafana-deployment.yaml b/grafana-deployment.yaml new file mode 100644 index 0000000..5034202 --- /dev/null +++ b/grafana-deployment.yaml @@ -0,0 +1,30 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: grafana +spec: + replicas: 1 + selector: + matchLabels: + app: grafana + template: + metadata: + labels: + app: grafana + spec: + containers: + - name: grafana + image: grafana/grafana + ports: + - containerPort: 3000 +--- +apiVersion: v1 +kind: Service +metadata: + name: grafana +spec: + type: LoadBalancer + ports: + - port: 3000 + selector: + app: grafana diff --git a/prometheus-development.yaml b/prometheus-development.yaml new file mode 100644 index 0000000..e49eabd --- /dev/null +++ b/prometheus-development.yaml @@ -0,0 +1,38 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + name: prometheus +spec: + replicas: 1 + selector: + matchLabels: + app: prometheus + template: + metadata: + labels: + app: prometheus + spec: + containers: + - name: prometheus + image: prom/prometheus + ports: + - containerPort: 9090 + volumeMounts: + - name: prometheus-config + mountPath: /etc/prometheus/ + subPath: prometheus.yml + volumes: + - name: prometheus-config + configMap: + name: prometheus-config +--- +apiVersion: v1 +kind: Service +metadata: + name: prometheus +spec: + type: LoadBalancer + ports: + - port: 9090 + selector: + app: prometheus diff --git a/prometheus.yml b/prometheus.yml new file mode 100644 index 0000000..3ee0fde --- /dev/null +++ b/prometheus.yml @@ -0,0 +1,8 @@ +global: + scrape_interval: 15s + +scrape_configs: + - job_name: 'snackscription-review' + metrics_path: '/actuator/prometheus' + static_configs: + - targets: ['snackscription-review-service:8080'] diff --git a/service.yaml b/service.yaml new file mode 100644 index 0000000..25b65be --- /dev/null +++ b/service.yaml @@ -0,0 +1,11 @@ +apiVersion: v1 +kind: Service +metadata: + name: snackscription-review-service +spec: + type: LoadBalancer + ports: + - port: 80 + targetPort: 8080 + selector: + app: snackscription-review diff --git a/src/main/java/snackscription/review/ReviewAppConfig.java b/src/main/java/snackscription/review/ReviewAppConfig.java new file mode 100644 index 0000000..e3519e7 --- /dev/null +++ b/src/main/java/snackscription/review/ReviewAppConfig.java @@ -0,0 +1,14 @@ +package snackscription.review; + +import io.micrometer.core.instrument.MeterRegistry; +import io.micrometer.core.instrument.composite.CompositeMeterRegistry; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; + +@Configuration +public class ReviewAppConfig { + @Bean + public MeterRegistry getMeterRegistry() { + return new CompositeMeterRegistry(); + } +} diff --git a/src/main/java/snackscription/review/service/ReviewService.java b/src/main/java/snackscription/review/service/ReviewService.java index 64462a3..b665e00 100644 --- a/src/main/java/snackscription/review/service/ReviewService.java +++ b/src/main/java/snackscription/review/service/ReviewService.java @@ -2,7 +2,10 @@ import java.util.List; import java.util.Optional; +import java.util.concurrent.atomic.AtomicInteger; +import io.micrometer.core.instrument.composite.CompositeMeterRegistry; +import org.springframework.stereotype.Component; import org.springframework.stereotype.Service; import snackscription.review.exception.InvalidStateException; @@ -13,8 +16,10 @@ import snackscription.review.repository.ReviewRepository; @Service +@Component public class ReviewService { private ReviewRepository reviewRepository; + private AtomicInteger activeUsers = null; public ReviewService (ReviewRepository reviewRepository) { this.reviewRepository = reviewRepository; From eefe1046b45fa46251194a448763f10315f6ede5 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 16:02:14 +0700 Subject: [PATCH 09/20] Add Prometheus and Grafana configuration --- .monitoring/docker-compose.yml | 19 ++++++++++ .../provisioning/datasources/datasources.yml | 7 ++++ .monitoring/prometheus/prometheus.yml | 8 ++++ application.yml | 5 +++ build.gradle.kts | 2 +- grafana-deployment.yaml | 30 --------------- prometheus-development.yaml | 38 ------------------- prometheus.yml | 8 ---- .../review/controller/ReviewController.java | 2 - .../review/service/ReviewService.java | 1 - 10 files changed, 40 insertions(+), 80 deletions(-) create mode 100644 .monitoring/docker-compose.yml create mode 100644 .monitoring/grafana/provisioning/datasources/datasources.yml create mode 100644 .monitoring/prometheus/prometheus.yml create mode 100644 application.yml delete mode 100644 grafana-deployment.yaml delete mode 100644 prometheus-development.yaml delete mode 100644 prometheus.yml diff --git a/.monitoring/docker-compose.yml b/.monitoring/docker-compose.yml new file mode 100644 index 0000000..87b2c31 --- /dev/null +++ b/.monitoring/docker-compose.yml @@ -0,0 +1,19 @@ +version: '3.7' + +services: + prometheus: + image: prom/prometheus:v2.44.0 + container_name: prometheus + ports: + - "9090:9090" + volumes: + - ./prometheus/prometheus.yml:/etc/prometheus/prometheus.yml + + grafana: + image: grafana/grafana:9.5.2 + container_name: grafana + ports: + - "3000:3000" + restart: unless-stopped + volumes: + - ./grafana/provisioning/datasources:/etc/grafana/provisioning/datasources diff --git a/.monitoring/grafana/provisioning/datasources/datasources.yml b/.monitoring/grafana/provisioning/datasources/datasources.yml new file mode 100644 index 0000000..8d9f9d8 --- /dev/null +++ b/.monitoring/grafana/provisioning/datasources/datasources.yml @@ -0,0 +1,7 @@ +apiVersion: 1 +datasources: + - name: Prometheus + type: prometheus + access: proxy + url: http://prometheus:9090 + isDefault: true \ No newline at end of file diff --git a/.monitoring/prometheus/prometheus.yml b/.monitoring/prometheus/prometheus.yml new file mode 100644 index 0000000..94f2cfe --- /dev/null +++ b/.monitoring/prometheus/prometheus.yml @@ -0,0 +1,8 @@ +scrape_configs: + - job_name: 'MyAppMetrics' + metrics_path: '/actuator/prometheus' + scrape_interval: 3s + static_configs: + - targets: ['host.docker.internal:8080'] + labels: + application: 'Snackscription Review' \ No newline at end of file diff --git a/application.yml b/application.yml new file mode 100644 index 0000000..3feefab --- /dev/null +++ b/application.yml @@ -0,0 +1,5 @@ +management: + endpoints: + web: + exposure: + include: [ "prometheus" ] diff --git a/build.gradle.kts b/build.gradle.kts index 23ef6a4..01cc3e8 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -27,10 +27,10 @@ dependencies { implementation("org.springframework.boot:spring-boot-starter-thymeleaf") implementation("org.springframework.boot:spring-boot-starter-web") implementation("org.springframework.boot:spring-boot-starter-actuator") - implementation("io.micrometer:micrometer-registry-prometheus") compileOnly("org.projectlombok:lombok") developmentOnly("org.springframework.boot:spring-boot-devtools") runtimeOnly("org.postgresql:postgresql") + runtimeOnly("io.micrometer:micrometer-registry-prometheus") annotationProcessor("org.springframework.boot:spring-boot-configuration-processor") annotationProcessor("org.projectlombok:lombok") testImplementation("org.springframework.boot:spring-boot-starter-test") diff --git a/grafana-deployment.yaml b/grafana-deployment.yaml deleted file mode 100644 index 5034202..0000000 --- a/grafana-deployment.yaml +++ /dev/null @@ -1,30 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: grafana -spec: - replicas: 1 - selector: - matchLabels: - app: grafana - template: - metadata: - labels: - app: grafana - spec: - containers: - - name: grafana - image: grafana/grafana - ports: - - containerPort: 3000 ---- -apiVersion: v1 -kind: Service -metadata: - name: grafana -spec: - type: LoadBalancer - ports: - - port: 3000 - selector: - app: grafana diff --git a/prometheus-development.yaml b/prometheus-development.yaml deleted file mode 100644 index e49eabd..0000000 --- a/prometheus-development.yaml +++ /dev/null @@ -1,38 +0,0 @@ -apiVersion: apps/v1 -kind: Deployment -metadata: - name: prometheus -spec: - replicas: 1 - selector: - matchLabels: - app: prometheus - template: - metadata: - labels: - app: prometheus - spec: - containers: - - name: prometheus - image: prom/prometheus - ports: - - containerPort: 9090 - volumeMounts: - - name: prometheus-config - mountPath: /etc/prometheus/ - subPath: prometheus.yml - volumes: - - name: prometheus-config - configMap: - name: prometheus-config ---- -apiVersion: v1 -kind: Service -metadata: - name: prometheus -spec: - type: LoadBalancer - ports: - - port: 9090 - selector: - app: prometheus diff --git a/prometheus.yml b/prometheus.yml deleted file mode 100644 index 3ee0fde..0000000 --- a/prometheus.yml +++ /dev/null @@ -1,8 +0,0 @@ -global: - scrape_interval: 15s - -scrape_configs: - - job_name: 'snackscription-review' - metrics_path: '/actuator/prometheus' - static_configs: - - targets: ['snackscription-review-service:8080'] diff --git a/src/main/java/snackscription/review/controller/ReviewController.java b/src/main/java/snackscription/review/controller/ReviewController.java index d126faa..bd802cd 100644 --- a/src/main/java/snackscription/review/controller/ReviewController.java +++ b/src/main/java/snackscription/review/controller/ReviewController.java @@ -13,8 +13,6 @@ @RestController @RequestMapping("/reviews") public class ReviewController { - - private ReviewService reviewService; public ReviewController(ReviewService reviewService) { diff --git a/src/main/java/snackscription/review/service/ReviewService.java b/src/main/java/snackscription/review/service/ReviewService.java index b665e00..31a94c5 100644 --- a/src/main/java/snackscription/review/service/ReviewService.java +++ b/src/main/java/snackscription/review/service/ReviewService.java @@ -19,7 +19,6 @@ @Component public class ReviewService { private ReviewRepository reviewRepository; - private AtomicInteger activeUsers = null; public ReviewService (ReviewRepository reviewRepository) { this.reviewRepository = reviewRepository; From 130fa41cca4c54eaae4033be3ffbacf5cb2631b6 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 18:52:02 +0700 Subject: [PATCH 10/20] [FIX] fix Dockerfile --- Dockerfile | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Dockerfile b/Dockerfile index 516b0e1..db9b952 100644 --- a/Dockerfile +++ b/Dockerfile @@ -10,6 +10,7 @@ ENV JDBC_DATABASE_URL ${JDBC_DATABASE_URL} ENV JDBC_DATABASE_USERNAME ${JDBC_DATABASE_USERNAME} WORKDIR /app -COPY build/libs/review-0.0.1-SNAPSHOT.jar /app/review-0.0.1-SNAPSHOT.jar +COPY ./review-0.0.1-SNAPSHOT.jar /app +RUN ls -la EXPOSE 8080 CMD ["java","-jar","review-0.0.1-SNAPSHOT.jar"] \ No newline at end of file From 26bc3056ee429b61f8a378917516e796e1bd7e8d Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 19:11:13 +0700 Subject: [PATCH 11/20] Add SonarCloud analysis workflow and configuration files --- .github/workflows/sonarcloud.yml | 68 ++++++++++++++++++++++++++++++++ sonar-project.properties | 6 +++ 2 files changed, 74 insertions(+) create mode 100644 .github/workflows/sonarcloud.yml create mode 100644 sonar-project.properties diff --git a/.github/workflows/sonarcloud.yml b/.github/workflows/sonarcloud.yml new file mode 100644 index 0000000..e9464da --- /dev/null +++ b/.github/workflows/sonarcloud.yml @@ -0,0 +1,68 @@ +# This workflow uses actions that are not certified by GitHub. +# They are provided by a third-party and are governed by +# separate terms of service, privacy policy, and support +# documentation. + +# This workflow helps you trigger a SonarCloud analysis of your code and populates +# GitHub Code Scanning alerts with the vulnerabilities found. +# Free for open source project. + +# 1. Login to SonarCloud.io using your GitHub account + +# 2. Import your project on SonarCloud +# * Add your GitHub organization first, then add your repository as a new project. +# * Please note that many languages are eligible for automatic analysis, +# which means that the analysis will start automatically without the need to set up GitHub Actions. +# * This behavior can be changed in Administration > Analysis Method. +# +# 3. Follow the SonarCloud in-product tutorial +# * a. Copy/paste the Project Key and the Organization Key into the args parameter below +# (You'll find this information in SonarCloud. Click on "Information" at the bottom left) +# +# * b. Generate a new token and add it to your Github repository's secrets using the name SONAR_TOKEN +# (On SonarCloud, click on your avatar on top-right > My account > Security +# or go directly to https://sonarcloud.io/account/security/) + +# Feel free to take a look at our documentation (https://docs.sonarcloud.io/getting-started/github/) +# or reach out to our community forum if you need some help (https://community.sonarsource.com/c/help/sc/9) + +name: SonarCloud analysis + +on: + push: + branches: [ "main" ] + pull_request: + branches: [ "main" ] + workflow_dispatch: + +permissions: + pull-requests: read # allows SonarCloud to decorate PRs with analysis results + +jobs: + Analysis: + runs-on: ubuntu-latest + + steps: + - name: Analyze with SonarCloud + + # You can pin the exact commit or the version. + # uses: SonarSource/sonarcloud-github-action@de2e56b42aa84d0b1c5b622644ac17e505c9a049 + uses: SonarSource/sonarcloud-github-action@de2e56b42aa84d0b1c5b622644ac17e505c9a049 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} # Needed to get PR information + SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} # Generate a token on Sonarcloud.io, add it to the secrets of this repo with the name SONAR_TOKEN (Settings > Secrets > Actions > add new repository secret) + with: + # Additional arguments for the sonarcloud scanner + args: + # Unique keys of your project and organization. You can find them in SonarCloud > Information (bottom-left menu) + # mandatory + -Dsonar.projectKey= + -Dsonar.organization= + # Comma-separated paths to directories containing main source files. + #-Dsonar.sources= # optional, default is project base directory + # When you need the analysis to take place in a directory other than the one from which it was launched + #-Dsonar.projectBaseDir= # optional, default is . + # Comma-separated paths to directories containing test source files. + #-Dsonar.tests= # optional. For more info about Code Coverage, please refer to https://docs.sonarcloud.io/enriching/test-coverage/overview/ + # Adds more detail to both client and server-side analysis logs, activating DEBUG mode for the scanner, and adding client-side environment variables and system properties to the server-side log of analysis report processing. + #-Dsonar.verbose= # optional, default is false diff --git a/sonar-project.properties b/sonar-project.properties new file mode 100644 index 0000000..83364ee --- /dev/null +++ b/sonar-project.properties @@ -0,0 +1,6 @@ +sonar.organization=adpro-c11 +sonar.projectKey=ADPRO-C11_snackscription-review + +# relative paths to source directories. More details and properties are described +# in https://sonarcloud.io/documentation/project-administration/narrowing-the-focus/ +sonar.sources=. \ No newline at end of file From 55c1249faef3c2e5a9992221b17db97b3faceb29 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 19:40:38 +0700 Subject: [PATCH 12/20] chore: Update SonarCloud workflow to include dev branch --- .github/workflows/sonarcloud.yml | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/sonarcloud.yml b/.github/workflows/sonarcloud.yml index e9464da..9069b9c 100644 --- a/.github/workflows/sonarcloud.yml +++ b/.github/workflows/sonarcloud.yml @@ -30,9 +30,9 @@ name: SonarCloud analysis on: push: - branches: [ "main" ] + branches: [ "main", "dev" ] pull_request: - branches: [ "main" ] + branches: [ "main", "dev" ] workflow_dispatch: permissions: From f3c9895a82cd169587841027d1ab61c4f0ca349b Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 19:51:33 +0700 Subject: [PATCH 13/20] [REFACTOR] Rename variables and improve code readability --- .github/workflows/sonarcloud.yml | 4 +- .../review/controller/ReviewController.java | 18 ++- .../controller/ReviewControllerTest.java | 134 ------------------ .../review/service/ReviewServiceTest.java | 2 - 4 files changed, 13 insertions(+), 145 deletions(-) diff --git a/.github/workflows/sonarcloud.yml b/.github/workflows/sonarcloud.yml index 9069b9c..82e63c3 100644 --- a/.github/workflows/sonarcloud.yml +++ b/.github/workflows/sonarcloud.yml @@ -56,8 +56,8 @@ jobs: args: # Unique keys of your project and organization. You can find them in SonarCloud > Information (bottom-left menu) # mandatory - -Dsonar.projectKey= - -Dsonar.organization= + -Dsonar.projectKey=ADPRO-C11_snackscription-review + -Dsonar.organization=adpro-c11 # Comma-separated paths to directories containing main source files. #-Dsonar.sources= # optional, default is project base directory # When you need the analysis to take place in a directory other than the one from which it was launched diff --git a/src/main/java/snackscription/review/controller/ReviewController.java b/src/main/java/snackscription/review/controller/ReviewController.java index bd802cd..4342da7 100644 --- a/src/main/java/snackscription/review/controller/ReviewController.java +++ b/src/main/java/snackscription/review/controller/ReviewController.java @@ -15,6 +15,10 @@ public class ReviewController { private ReviewService reviewService; + public static final String BODY_AUTHOR = "author"; + public static final String BODY_CONTENT = "content"; + public static final String BODY_RATING = "rating"; + public ReviewController(ReviewService reviewService) { this.reviewService = reviewService; } @@ -27,9 +31,9 @@ public ResponseEntity reviewPage() { @PostMapping("/subscription-boxes/{subsbox}") public ResponseEntity createSubsboxReview(@RequestBody Map body, @PathVariable String subsbox) { try { - String author = body.get("author"); - int rating = Integer.parseInt(body.get("rating")); - String content = body.get("content"); + String author = body.get(BODY_AUTHOR); + int rating = Integer.parseInt(body.get(BODY_RATING)); + String content = body.get(BODY_CONTENT); Review review = reviewService.createReview(rating, content, subsbox, author); return new ResponseEntity<>(review, HttpStatus.CREATED); @@ -51,7 +55,7 @@ public ResponseEntity> getPublicSubsboxReview(@PathVariable String @GetMapping("/subscription-boxes/{subsbox}/users/{user}") public ResponseEntity getSelfSubsboxReview(@RequestBody Map body, @PathVariable String subsbox, @PathVariable String user) { try { - String sender = body.get("author"); // TODO: nanti pakai JWT token untuk ambil sendernya + String sender = body.get(BODY_AUTHOR); if (!authenticate(sender, user)) { return new ResponseEntity<>(HttpStatus.UNAUTHORIZED); } @@ -65,13 +69,13 @@ public ResponseEntity getSelfSubsboxReview(@RequestBody Map editReview(@RequestBody Map body, @PathVariable String subsbox, @PathVariable String user) { try { - String sender = body.get("author"); // TODO: nanti pakai JWT token untuk ambil sendernya + String sender = body.get(BODY_AUTHOR); if (!authenticate(sender, user)) { return new ResponseEntity<>(HttpStatus.UNAUTHORIZED); } - int rating = Integer.parseInt(body.get("rating")); - String content = body.get("content"); + int rating = Integer.parseInt(body.get(BODY_RATING)); + String content = body.get(BODY_CONTENT); Review review = reviewService.editReview(rating, content, subsbox, user); return new ResponseEntity<>(review, HttpStatus.OK); diff --git a/src/test/java/snackscription/review/controller/ReviewControllerTest.java b/src/test/java/snackscription/review/controller/ReviewControllerTest.java index bf0ccca..8e3035e 100644 --- a/src/test/java/snackscription/review/controller/ReviewControllerTest.java +++ b/src/test/java/snackscription/review/controller/ReviewControllerTest.java @@ -146,44 +146,6 @@ public void readSelfSubscriptionBoxReview() throws Exception { verify(reviewService).getReview(subsbox, author); } -// @Test -// public void testEditSelfSubscriptionBoxReview() throws Exception { -// Review review = reviews.getFirst(); -// String subsboxId = review.getSubsbox(); -// String userId = review.getAuthor(); -// -// int newRating = 4; -// String newContent = "Awikwok"; -// when(reviewService.editReview(newRating, newContent, subsboxId, userId)).thenReturn(new Review(newRating, newContent, userId, subsboxId)); -// -// ResultActions result = mockMvc.perform(put("/reviews/subscription-boxes/{subscriptionBoxId}/users/self", subsboxId) -// .contentType(MediaType.APPLICATION_JSON) -// .content("{\"rating\": 4, \"content\": \"Awikwok\", \"userId\": \"user_123\"}")) -// .andExpect(status().isOk()) -// .andExpect(jsonPath("$.rating", is(newRating))) -// .andExpect(jsonPath("$.content", is(newContent))) -// .andExpect(jsonPath("$.userId", is(review.getAuthor()))) -// .andExpect(jsonPath("$.subscriptionBoxId", is(review.getSubsbox()))); -// -// verify(reviewService).editReview(newRating, newContent, subsboxId, userId); -// } -// -// @Test -// public void testDeleteSelfSubscriptionBoxReview() throws Exception { -// Review review = reviews.getFirst(); -// String subsboxId = review.getSubsbox(); -// String userId = review.getAuthor(); -// -// doNothing().when(reviewService).deleteReview(subsboxId, userId); -// -// ResultActions result = mockMvc.perform(delete("/subscription-boxes/{subscriptionBoxId}/users/self", subsboxId) -// .contentType(MediaType.APPLICATION_JSON) -// .content("{\"userId\": \"user_123\"}")) -// .andExpect(status().isNoContent()); -// -// verify(reviewService).deleteReview(subsboxId, userId); -// } - @Test public void testDeleteUserSubscriptionBoxReview() throws Exception { Review review = reviews.getFirst(); @@ -238,100 +200,4 @@ public void testRejectReview() throws Exception { verify(reviewService).approveReview(review.getSubsbox(), review.getAuthor()); } - -// @Test -// public void testGetAllSubscriptionBoxReview() { -// String subsboxId = "subsboxId"; - -// ArrayList reviews = new ArrayList<>(); -// reviews.add(new Review(5, "amazing", "user1", subsboxId)); -// reviews.add(new Review(4, "good", "user2", subsboxId)); - -// when(reviewService.testGetAllSubscriptionBoxReview(subsboxId)).thenReturn(reviews); - -// ResultActions result = mockMvc.perform(get("/api/subscription-boxes/{subsboxId}", subsboxId)) -// .andExpect(status().isOk()) -// .andExpect(jsonPath("$", hasSize(2))) -// .andExpect(jsonPath("$[0].rating", is(5))) -// .andExpect(jsonPath("$[0].content", is("amazing"))) -// .andExpect(jsonPath("$[0].userId", is("user1"))) -// .andExpect(jsonPath("$[0].subscriptionBoxId", is(subsboxId))) -// .andExpect(jsonPath("$[1].rating", is(4))) -// .andExpect(jsonPath("$[1].content", is("good"))) -// .andExpect(jsonPath("$[1].userId", is("user2"))) -// .andExpect(jsonPath("$[1].subscriptionBoxId", is(subsboxId))); - -// verify(reviewService).testGetAllSubscriptionBoxReview(subsboxId); -// } - -// @Test -// public void testGetById() throws Exception { -// Review review = new Review( -// 5, "amazing", "user1", "subsboxId" -// ); -// String reviewId = review.getId(); - -// when(reviewService.findById(reviewId)).thenReturn(review); - -// ResultActions result = mockMvc.perform(get("/api/reviews/{reviewId}", reviewId)) -// .andExpect(status().isOk()) -// .andExpect(jsonPath("$.rating", is(5))) -// .andExpect(jsonPath("$.content", is("amazing"))) -// .andExpect(jsonPath("$.userId", is("user1"))) -// .andExpect(jsonPath("$.subscriptionBoxId", is("subsboxId"))); - -// verify(reviewService).findById(reviewId); -// } - -// @Test -// public void testGetBySubscriptionBoxId() throws Exception { -// List curReviews = new ArrayList<>(); - -// String subscriptionBoxId = this.reviews.getFirst().getSubsbox(); -// for (Review review : this.reviews) { -// if (review.getSubsbox().equals(subscriptionBoxId)) { -// curReviews.add(review); -// } -// } - -// when(reviewService.findBySubscriptionBoxId(subscriptionBoxId)).thenReturn(curReviews); - -// String result = mockMvc.perform(get("/api/subscription-boxes/{subscriptionBoxId}", subscriptionBoxId)) -// .andExpect(status().isOk()) -// .andExpect(jsonPath("$", hasSize(curReviews.size()))) -// .andReturn() -// .getResponse() -// .getContentAsString(); - -// List foundReviews = new ArrayList(); -// for (int i=0; i cmp = new Comparator() { -// @Override -// public int compare(Review o1, Review o2) { -// return o1.getAuthor().compareTo(o2.getAuthor()); -// } -// }; - -// curReviews.sort(cmp); -// foundReviews.sort(cmp); - -// for (int i=0; i curReviews = new ArrayList<>(); String subscriptionBoxId = this.reviews.getFirst().getSubsbox(); From 5b315acf057b702b23763632af99a3aa85259dde Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 19:56:23 +0700 Subject: [PATCH 14/20] [FIX] Fix sonarcloud workflow --- .github/workflows/sonarcloud.yml | 27 ++++++++++++++++++++------- 1 file changed, 20 insertions(+), 7 deletions(-) diff --git a/.github/workflows/sonarcloud.yml b/.github/workflows/sonarcloud.yml index 82e63c3..3fc2599 100644 --- a/.github/workflows/sonarcloud.yml +++ b/.github/workflows/sonarcloud.yml @@ -30,10 +30,8 @@ name: SonarCloud analysis on: push: - branches: [ "main", "dev" ] - pull_request: - branches: [ "main", "dev" ] - workflow_dispatch: + branches: + - "**" permissions: pull-requests: read # allows SonarCloud to decorate PRs with analysis results @@ -43,11 +41,23 @@ jobs: runs-on: ubuntu-latest steps: - - name: Analyze with SonarCloud + - name: Check out the Git repository + uses: actions/checkout@v4 + + - name: Set up Java toolchain + uses: actions/setup-java@v4 + with: + distribution: "temurin" + java-version: "21" + cache: "gradle" + - name: Build + run: ./gradlew build + + - name: Analyze with SonarCloud # You can pin the exact commit or the version. # uses: SonarSource/sonarcloud-github-action@de2e56b42aa84d0b1c5b622644ac17e505c9a049 - uses: SonarSource/sonarcloud-github-action@de2e56b42aa84d0b1c5b622644ac17e505c9a049 + uses: SonarSource/sonarcloud-github-action@master env: GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} # Needed to get PR information SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} # Generate a token on Sonarcloud.io, add it to the secrets of this repo with the name SONAR_TOKEN (Settings > Secrets > Actions > add new repository secret) @@ -58,6 +68,9 @@ jobs: # mandatory -Dsonar.projectKey=ADPRO-C11_snackscription-review -Dsonar.organization=adpro-c11 + -Dsonar.java.binaries=. + -Dsonar.sources=src/main/java + -Dsonar.tests=src/test/java # Comma-separated paths to directories containing main source files. #-Dsonar.sources= # optional, default is project base directory # When you need the analysis to take place in a directory other than the one from which it was launched @@ -65,4 +78,4 @@ jobs: # Comma-separated paths to directories containing test source files. #-Dsonar.tests= # optional. For more info about Code Coverage, please refer to https://docs.sonarcloud.io/enriching/test-coverage/overview/ # Adds more detail to both client and server-side analysis logs, activating DEBUG mode for the scanner, and adding client-side environment variables and system properties to the server-side log of analysis report processing. - #-Dsonar.verbose= # optional, default is false + #-Dsonar.verbose= # optional, default is false \ No newline at end of file From 58fcb8c9b1c0a5fdb1e73a7054c5ba0e9758c148 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 20:03:26 +0700 Subject: [PATCH 15/20] [FIX] Add test coverage --- .github/workflows/sonarcloud.yml | 13 ------------- build.gradle.kts | 3 ++- 2 files changed, 2 insertions(+), 14 deletions(-) diff --git a/.github/workflows/sonarcloud.yml b/.github/workflows/sonarcloud.yml index 3fc2599..5e50c08 100644 --- a/.github/workflows/sonarcloud.yml +++ b/.github/workflows/sonarcloud.yml @@ -41,19 +41,6 @@ jobs: runs-on: ubuntu-latest steps: - - name: Check out the Git repository - uses: actions/checkout@v4 - - - name: Set up Java toolchain - uses: actions/setup-java@v4 - with: - distribution: "temurin" - java-version: "21" - cache: "gradle" - - - name: Build - run: ./gradlew build - - name: Analyze with SonarCloud # You can pin the exact commit or the version. # uses: SonarSource/sonarcloud-github-action@de2e56b42aa84d0b1c5b622644ac17e505c9a049 diff --git a/build.gradle.kts b/build.gradle.kts index 01cc3e8..1fc1f0a 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -50,7 +50,8 @@ tasks.jacocoTestReport { })) dependsOn(tasks.test) reports { - xml.required.set(false) + xml.required.set(true) + html.required.set(true) csv.required.set(false) html.outputLocation.set(layout.buildDirectory.dir("jacocoHtml")) } From ad3688228ba031097fe0f7245037698061e399bb Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 20:06:35 +0700 Subject: [PATCH 16/20] [FIX] Update workflow --- .github/workflows/ci.yml | 33 +++++++++++++--- .github/workflows/sonarcloud.yml | 68 -------------------------------- 2 files changed, 27 insertions(+), 74 deletions(-) delete mode 100644 .github/workflows/sonarcloud.yml diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index de84db9..b781189 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -3,12 +3,7 @@ name: Java CI Pipeline on: push: branches: - - main - - dev - pull_request: - branches: - - main - - dev + - "**" workflow_dispatch: jobs: @@ -90,3 +85,29 @@ jobs: env: PRODUCTION: test # (Optional) Add steps for generating coverage report and other post-test tasks + + - name: Analyze with SonarCloud + # You can pin the exact commit or the version. + # uses: SonarSource/sonarcloud-github-action@de2e56b42aa84d0b1c5b622644ac17e505c9a049 + uses: SonarSource/sonarcloud-github-action@master + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} # Needed to get PR information + SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} # Generate a token on Sonarcloud.io, add it to the secrets of this repo with the name SONAR_TOKEN (Settings > Secrets > Actions > add new repository secret) + with: + # Additional arguments for the sonarcloud scanner + args: + # Unique keys of your project and organization. You can find them in SonarCloud > Information (bottom-left menu) + # mandatory + -Dsonar.projectKey=ADPRO-C11_snackscription-review + -Dsonar.organization=adpro-c11 + -Dsonar.java.binaries=. + -Dsonar.sources=src/main/java + -Dsonar.tests=src/test/java + # Comma-separated paths to directories containing main source files. + #-Dsonar.sources= # optional, default is project base directory + # When you need the analysis to take place in a directory other than the one from which it was launched + #-Dsonar.projectBaseDir= # optional, default is . + # Comma-separated paths to directories containing test source files. + #-Dsonar.tests= # optional. For more info about Code Coverage, please refer to https://docs.sonarcloud.io/enriching/test-coverage/overview/ + # Adds more detail to both client and server-side analysis logs, activating DEBUG mode for the scanner, and adding client-side environment variables and system properties to the server-side log of analysis report processing. + #-Dsonar.verbose= # optional, default is false \ No newline at end of file diff --git a/.github/workflows/sonarcloud.yml b/.github/workflows/sonarcloud.yml deleted file mode 100644 index 5e50c08..0000000 --- a/.github/workflows/sonarcloud.yml +++ /dev/null @@ -1,68 +0,0 @@ -# This workflow uses actions that are not certified by GitHub. -# They are provided by a third-party and are governed by -# separate terms of service, privacy policy, and support -# documentation. - -# This workflow helps you trigger a SonarCloud analysis of your code and populates -# GitHub Code Scanning alerts with the vulnerabilities found. -# Free for open source project. - -# 1. Login to SonarCloud.io using your GitHub account - -# 2. Import your project on SonarCloud -# * Add your GitHub organization first, then add your repository as a new project. -# * Please note that many languages are eligible for automatic analysis, -# which means that the analysis will start automatically without the need to set up GitHub Actions. -# * This behavior can be changed in Administration > Analysis Method. -# -# 3. Follow the SonarCloud in-product tutorial -# * a. Copy/paste the Project Key and the Organization Key into the args parameter below -# (You'll find this information in SonarCloud. Click on "Information" at the bottom left) -# -# * b. Generate a new token and add it to your Github repository's secrets using the name SONAR_TOKEN -# (On SonarCloud, click on your avatar on top-right > My account > Security -# or go directly to https://sonarcloud.io/account/security/) - -# Feel free to take a look at our documentation (https://docs.sonarcloud.io/getting-started/github/) -# or reach out to our community forum if you need some help (https://community.sonarsource.com/c/help/sc/9) - -name: SonarCloud analysis - -on: - push: - branches: - - "**" - -permissions: - pull-requests: read # allows SonarCloud to decorate PRs with analysis results - -jobs: - Analysis: - runs-on: ubuntu-latest - - steps: - - name: Analyze with SonarCloud - # You can pin the exact commit or the version. - # uses: SonarSource/sonarcloud-github-action@de2e56b42aa84d0b1c5b622644ac17e505c9a049 - uses: SonarSource/sonarcloud-github-action@master - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} # Needed to get PR information - SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} # Generate a token on Sonarcloud.io, add it to the secrets of this repo with the name SONAR_TOKEN (Settings > Secrets > Actions > add new repository secret) - with: - # Additional arguments for the sonarcloud scanner - args: - # Unique keys of your project and organization. You can find them in SonarCloud > Information (bottom-left menu) - # mandatory - -Dsonar.projectKey=ADPRO-C11_snackscription-review - -Dsonar.organization=adpro-c11 - -Dsonar.java.binaries=. - -Dsonar.sources=src/main/java - -Dsonar.tests=src/test/java - # Comma-separated paths to directories containing main source files. - #-Dsonar.sources= # optional, default is project base directory - # When you need the analysis to take place in a directory other than the one from which it was launched - #-Dsonar.projectBaseDir= # optional, default is . - # Comma-separated paths to directories containing test source files. - #-Dsonar.tests= # optional. For more info about Code Coverage, please refer to https://docs.sonarcloud.io/enriching/test-coverage/overview/ - # Adds more detail to both client and server-side analysis logs, activating DEBUG mode for the scanner, and adding client-side environment variables and system properties to the server-side log of analysis report processing. - #-Dsonar.verbose= # optional, default is false \ No newline at end of file From 0bcb63d5d90599e9a29aa450ebe77eb0a9f8404e Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sat, 18 May 2024 20:18:24 +0700 Subject: [PATCH 17/20] [FIX] Remove sonarcloud workflow --- .github/workflows/ci.yml | 28 +--------------------------- sonar-project.properties | 6 ------ 2 files changed, 1 insertion(+), 33 deletions(-) delete mode 100644 sonar-project.properties diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index b781189..050ab36 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -84,30 +84,4 @@ jobs: ./gradlew jacocoTestReport env: PRODUCTION: test - # (Optional) Add steps for generating coverage report and other post-test tasks - - - name: Analyze with SonarCloud - # You can pin the exact commit or the version. - # uses: SonarSource/sonarcloud-github-action@de2e56b42aa84d0b1c5b622644ac17e505c9a049 - uses: SonarSource/sonarcloud-github-action@master - env: - GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} # Needed to get PR information - SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} # Generate a token on Sonarcloud.io, add it to the secrets of this repo with the name SONAR_TOKEN (Settings > Secrets > Actions > add new repository secret) - with: - # Additional arguments for the sonarcloud scanner - args: - # Unique keys of your project and organization. You can find them in SonarCloud > Information (bottom-left menu) - # mandatory - -Dsonar.projectKey=ADPRO-C11_snackscription-review - -Dsonar.organization=adpro-c11 - -Dsonar.java.binaries=. - -Dsonar.sources=src/main/java - -Dsonar.tests=src/test/java - # Comma-separated paths to directories containing main source files. - #-Dsonar.sources= # optional, default is project base directory - # When you need the analysis to take place in a directory other than the one from which it was launched - #-Dsonar.projectBaseDir= # optional, default is . - # Comma-separated paths to directories containing test source files. - #-Dsonar.tests= # optional. For more info about Code Coverage, please refer to https://docs.sonarcloud.io/enriching/test-coverage/overview/ - # Adds more detail to both client and server-side analysis logs, activating DEBUG mode for the scanner, and adding client-side environment variables and system properties to the server-side log of analysis report processing. - #-Dsonar.verbose= # optional, default is false \ No newline at end of file + # (Optional) Add steps for generating coverage report and other post-test tasks \ No newline at end of file diff --git a/sonar-project.properties b/sonar-project.properties deleted file mode 100644 index 83364ee..0000000 --- a/sonar-project.properties +++ /dev/null @@ -1,6 +0,0 @@ -sonar.organization=adpro-c11 -sonar.projectKey=ADPRO-C11_snackscription-review - -# relative paths to source directories. More details and properties are described -# in https://sonarcloud.io/documentation/project-administration/narrowing-the-focus/ -sonar.sources=. \ No newline at end of file From 4bc3c8f42e51632739b897d4bed9b82b7ca6cd27 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sun, 19 May 2024 19:34:28 +0700 Subject: [PATCH 18/20] Revert "Merge pull request #12 from ADPRO-C11/async" This reverts commit 697428ab5c0148081c460742bfc80746f7c5d590, reversing changes made to 130fa41cca4c54eaae4033be3ffbacf5cb2631b6. The Sonarcloud setting is not working yet, fixing it soon. --- .github/workflows/cd.yml | 10 ---------- .../review/ReviewApplication.java | 18 ++---------------- .../review/service/ReviewService.java | 10 ---------- .../service/SentimentAnalysisService.java | 8 -------- .../review/service/ReviewServiceTest.java | 5 ----- 5 files changed, 2 insertions(+), 49 deletions(-) delete mode 100644 src/main/java/snackscription/review/service/SentimentAnalysisService.java diff --git a/.github/workflows/cd.yml b/.github/workflows/cd.yml index 40fcb7c..30e9471 100644 --- a/.github/workflows/cd.yml +++ b/.github/workflows/cd.yml @@ -45,16 +45,6 @@ jobs: name: Test runs-on: ubuntu-latest needs: build - services: - postgres: - image: postgres:latest - env: - POSTGRES_USER: postgres - POSTGRES_PASSWORD: postgres - POSTGRES_DB: snackscription_review - ports: - - 5432:5432 - options: --health-cmd pg_isready --health-interval 10s --health-timeout 5s --health-retries 5 steps: - name: Checkout Repository uses: actions/checkout@v4 diff --git a/src/main/java/snackscription/review/ReviewApplication.java b/src/main/java/snackscription/review/ReviewApplication.java index 4898e1d..ec64e1c 100644 --- a/src/main/java/snackscription/review/ReviewApplication.java +++ b/src/main/java/snackscription/review/ReviewApplication.java @@ -2,28 +2,14 @@ import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; -import org.springframework.context.annotation.Bean; -import org.springframework.scheduling.annotation.EnableAsync; -import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; -import java.util.concurrent.Executor; - -@EnableAsync @SpringBootApplication public class ReviewApplication { public static void main(String[] args) { SpringApplication.run(ReviewApplication.class, args); - } - @Bean - public Executor taskExecutor () { - ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor(); - executor.setCorePoolSize(1); - executor.setMaxPoolSize(1); - executor.setQueueCapacity(500); - executor.setThreadNamePrefix("GithubLookup-"); - executor.initialize(); - return executor; + } + } diff --git a/src/main/java/snackscription/review/service/ReviewService.java b/src/main/java/snackscription/review/service/ReviewService.java index 44883b9..31a94c5 100644 --- a/src/main/java/snackscription/review/service/ReviewService.java +++ b/src/main/java/snackscription/review/service/ReviewService.java @@ -2,9 +2,6 @@ import java.util.List; import java.util.Optional; -import java.util.concurrent.CompletableFuture; - -import org.springframework.scheduling.annotation.Async; import java.util.concurrent.atomic.AtomicInteger; import io.micrometer.core.instrument.composite.CompositeMeterRegistry; @@ -22,7 +19,6 @@ @Component public class ReviewService { private ReviewRepository reviewRepository; - private SentimentAnalysisService sentimentAnalysisService; public ReviewService (ReviewRepository reviewRepository) { this.reviewRepository = reviewRepository; @@ -80,12 +76,6 @@ public Review rejectReview(String subsbox, String user) throws Exception { return reviewRepository.save(review); } - @Async - public CompletableFuture analyzeSentimentAsync(String reviewText) { - String sentiment = sentimentAnalysisService.analyze(reviewText); - return CompletableFuture.completedFuture(sentiment); - } - public void deleteReview(String subsbox, String user) throws Exception { Review review = reviewRepository.findByIdSubsboxAndIdAuthor(subsbox, user); diff --git a/src/main/java/snackscription/review/service/SentimentAnalysisService.java b/src/main/java/snackscription/review/service/SentimentAnalysisService.java deleted file mode 100644 index b9610a8..0000000 --- a/src/main/java/snackscription/review/service/SentimentAnalysisService.java +++ /dev/null @@ -1,8 +0,0 @@ -package snackscription.review.service; - -public class SentimentAnalysisService { - public String analyze(String reviewText) { - return "positive"; - } - -} diff --git a/src/test/java/snackscription/review/service/ReviewServiceTest.java b/src/test/java/snackscription/review/service/ReviewServiceTest.java index 4fdf16d..6350528 100644 --- a/src/test/java/snackscription/review/service/ReviewServiceTest.java +++ b/src/test/java/snackscription/review/service/ReviewServiceTest.java @@ -179,10 +179,5 @@ public void assertEqualReview(Review review1, Review review2) { assertEquals(review1.getAuthor(), review2.getAuthor()); assertEquals(review1.getSubsbox(), review2.getSubsbox()); } - - @Test - public void analyzeSentimentAsyncTest() { - - } } From c4e50d3daee45f06c18df7c2cfdff95a18a9400c Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Mon, 20 May 2024 10:35:15 +0700 Subject: [PATCH 19/20] Configure prometheus to monitor deployed app --- .monitoring/prometheus/prometheus.yml | 7 +++++-- build.gradle.kts | 3 ++- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/.monitoring/prometheus/prometheus.yml b/.monitoring/prometheus/prometheus.yml index 94f2cfe..f454fff 100644 --- a/.monitoring/prometheus/prometheus.yml +++ b/.monitoring/prometheus/prometheus.yml @@ -1,8 +1,11 @@ scrape_configs: - - job_name: 'MyAppMetrics' + - job_name: 'Snackscription Metrics' metrics_path: '/actuator/prometheus' scrape_interval: 3s static_configs: - targets: ['host.docker.internal:8080'] labels: - application: 'Snackscription Review' \ No newline at end of file + application: 'Snackscription Review' + - targets: ['34.124.152.90'] + labels: + application: 'Snackscription Review (deployed)' \ No newline at end of file diff --git a/build.gradle.kts b/build.gradle.kts index 01cc3e8..0acc097 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -27,10 +27,11 @@ dependencies { implementation("org.springframework.boot:spring-boot-starter-thymeleaf") implementation("org.springframework.boot:spring-boot-starter-web") implementation("org.springframework.boot:spring-boot-starter-actuator") + implementation("io.micrometer:micrometer-registry-prometheus") + implementation("io.micrometer:micrometer-core") compileOnly("org.projectlombok:lombok") developmentOnly("org.springframework.boot:spring-boot-devtools") runtimeOnly("org.postgresql:postgresql") - runtimeOnly("io.micrometer:micrometer-registry-prometheus") annotationProcessor("org.springframework.boot:spring-boot-configuration-processor") annotationProcessor("org.projectlombok:lombok") testImplementation("org.springframework.boot:spring-boot-starter-test") From c8aac1a23a2cbb580af164a8ad11e553144fe7d0 Mon Sep 17 00:00:00 2001 From: asteriskzie Date: Sun, 26 May 2024 08:13:39 +0700 Subject: [PATCH 20/20] [FIX] Fix Sonar settings --- build.gradle.kts | 10 ++++++++++ gradle/gradle.properties | 5 +++++ 2 files changed, 15 insertions(+) create mode 100644 gradle/gradle.properties diff --git a/build.gradle.kts b/build.gradle.kts index 1fc1f0a..0572b5f 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -3,6 +3,7 @@ plugins { id("org.springframework.boot") version "3.2.5" id("io.spring.dependency-management") version "1.1.4" jacoco + id("org.sonarqube") version "4.4.1.3373" } group = "snackscription" @@ -12,6 +13,7 @@ java { sourceCompatibility = JavaVersion.VERSION_21 } + configurations { compileOnly { extendsFrom(configurations.annotationProcessor.get()) @@ -55,4 +57,12 @@ tasks.jacocoTestReport { csv.required.set(false) html.outputLocation.set(layout.buildDirectory.dir("jacocoHtml")) } +} + +sonar { + properties { + property("sonar.projectKey","ADPRO-C11_snackscription-review") + property("sonar.organization", "adpro-c11") + property("sonar.host.url", "https://sonarcloud.io") + } } \ No newline at end of file diff --git a/gradle/gradle.properties b/gradle/gradle.properties new file mode 100644 index 0000000..2b4c21a --- /dev/null +++ b/gradle/gradle.properties @@ -0,0 +1,5 @@ +systemProp.sonar.host.url=https://sonarcloud.io + +# Token generated from an account with 'Execute analysis' permission. +# It can also be set with the environment variable SONAR_TOKEN. +systemProp.sonar.token=${SONAR_TOKEN} \ No newline at end of file