| 1 | package edu.ucsb.cs156.example.controllers; | |
| 2 | ||
| 3 | import edu.ucsb.cs156.example.entities.Article; | |
| 4 | import edu.ucsb.cs156.example.errors.EntityNotFoundException; | |
| 5 | import edu.ucsb.cs156.example.repositories.ArticlesRepository; | |
| 6 | ||
| 7 | import io.swagger.v3.oas.annotations.Operation; | |
| 8 | import io.swagger.v3.oas.annotations.Parameter; | |
| 9 | import io.swagger.v3.oas.annotations.tags.Tag; | |
| 10 | import lombok.extern.slf4j.Slf4j; | |
| 11 | ||
| 12 | import com.fasterxml.jackson.core.JsonProcessingException; | |
| 13 | ||
| 14 | import org.springframework.beans.factory.annotation.Autowired; | |
| 15 | import org.springframework.format.annotation.DateTimeFormat; | |
| 16 | import org.springframework.http.HttpStatus; | |
| 17 | import org.springframework.security.access.prepost.PreAuthorize; | |
| 18 | import org.springframework.web.bind.annotation.DeleteMapping; | |
| 19 | import org.springframework.web.bind.annotation.GetMapping; | |
| 20 | import org.springframework.web.bind.annotation.PostMapping; | |
| 21 | import org.springframework.web.bind.annotation.PutMapping; | |
| 22 | import org.springframework.web.bind.annotation.RequestBody; | |
| 23 | import org.springframework.web.bind.annotation.RequestMapping; | |
| 24 | import org.springframework.web.bind.annotation.RequestParam; | |
| 25 | import org.springframework.web.bind.annotation.RestController; | |
| 26 | import org.springframework.web.server.ResponseStatusException; | |
| 27 | ||
| 28 | import jakarta.validation.Valid; | |
| 29 | ||
| 30 | import java.time.LocalDateTime; | |
| 31 | import java.time.format.DateTimeFormatter; | |
| 32 | import java.time.format.DateTimeParseException; | |
| 33 | import java.util.List; | |
| 34 | ||
| 35 | /** | |
| 36 | * This is a REST controller for Articles | |
| 37 | */ | |
| 38 | ||
| 39 | @Tag(name = "Articles") | |
| 40 | @RequestMapping("/api/articles") | |
| 41 | @RestController | |
| 42 | @Slf4j | |
| 43 | public class ArticlesController extends ApiController { | |
| 44 | ||
| 45 | @Autowired | |
| 46 | ArticlesRepository articlesRepository; | |
| 47 | ||
| 48 | /** | |
| 49 | * List all articles | |
| 50 | * | |
| 51 | * @return an iterable of Articles | |
| 52 | */ | |
| 53 | @Operation(summary= "List all articles") | |
| 54 | @PreAuthorize("hasRole('ROLE_USER')") | |
| 55 | @GetMapping("/all") | |
| 56 | public Iterable<Article> allArticles() { | |
| 57 |
1
1. allArticles : replaced return value with Collections.emptyList for edu/ucsb/cs156/example/controllers/ArticlesController::allArticles → KILLED |
return articlesRepository.findAll(); |
| 58 | } | |
| 59 | ||
| 60 | /** | |
| 61 | * Get a single article by id | |
| 62 | * | |
| 63 | * @param id the id of the article | |
| 64 | * @return an Article | |
| 65 | */ | |
| 66 | @Operation(summary= "Get a single article") | |
| 67 | @PreAuthorize("hasRole('ROLE_USER')") | |
| 68 | @GetMapping("") | |
| 69 | public Article getById( | |
| 70 | @Parameter(name="id") @RequestParam Long id) { | |
| 71 | Article article = articlesRepository.findById(id) | |
| 72 |
1
1. lambda$getById$0 : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::lambda$getById$0 → KILLED |
.orElseThrow(() -> new EntityNotFoundException(Article.class, id)); |
| 73 | ||
| 74 |
1
1. getById : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::getById → KILLED |
return article; |
| 75 | } | |
| 76 | ||
| 77 | /** | |
| 78 | * Create a new article | |
| 79 | * | |
| 80 | * @param title the title of the article | |
| 81 | * @param url the url of the article | |
| 82 | * @param explanation the explanation of the article | |
| 83 | * @param email the email of the person who added the article | |
| 84 | * @param dateAdded the date the article was added | |
| 85 | * @return the saved article | |
| 86 | */ | |
| 87 | @Operation(summary= "Create a new article") | |
| 88 | @PreAuthorize("hasRole('ROLE_ADMIN')") | |
| 89 | @PostMapping("/post") | |
| 90 | public Article postArticle( | |
| 91 | @Parameter(name="title") @RequestParam String title, | |
| 92 | @Parameter(name="url") @RequestParam String url, | |
| 93 | @Parameter(name="explanation") @RequestParam String explanation, | |
| 94 | @Parameter(name="email") @RequestParam String email, | |
| 95 | @Parameter(name="dateAdded") @RequestParam String dateAdded) | |
| 96 | throws JsonProcessingException { | |
| 97 | ||
| 98 | LocalDateTime dateTime; | |
| 99 | try { | |
| 100 | dateTime = parseDateTime(dateAdded); | |
| 101 | } catch (DateTimeParseException e) { | |
| 102 | throw new ResponseStatusException(HttpStatus.BAD_REQUEST, "Invalid date format. Use ISO date format (YYYY-MM-DD) or ISO date-time format (YYYY-MM-DDThh:mm:ss)"); | |
| 103 | } | |
| 104 | ||
| 105 | log.info("dateAdded={}", dateTime); | |
| 106 | ||
| 107 | Article article = Article.builder() | |
| 108 | .title(title) | |
| 109 | .url(url) | |
| 110 | .explanation(explanation) | |
| 111 | .email(email) | |
| 112 | .dateAdded(dateTime) | |
| 113 | .build(); | |
| 114 | ||
| 115 | Article savedArticle = articlesRepository.save(article); | |
| 116 | ||
| 117 |
1
1. postArticle : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::postArticle → KILLED |
return savedArticle; |
| 118 | } | |
| 119 | ||
| 120 | /** | |
| 121 | * Delete an Article | |
| 122 | * | |
| 123 | * @param id the id of the article to delete | |
| 124 | * @return a message indicating the article was deleted | |
| 125 | */ | |
| 126 | @Operation(summary= "Delete an Article") | |
| 127 | @PreAuthorize("hasRole('ROLE_ADMIN')") | |
| 128 | @DeleteMapping("") | |
| 129 | public Object deleteArticle( | |
| 130 | @Parameter(name="id") @RequestParam Long id) { | |
| 131 | Article article = articlesRepository.findById(id) | |
| 132 |
1
1. lambda$deleteArticle$1 : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::lambda$deleteArticle$1 → KILLED |
.orElseThrow(() -> new EntityNotFoundException(Article.class, id)); |
| 133 | ||
| 134 |
1
1. deleteArticle : removed call to edu/ucsb/cs156/example/repositories/ArticlesRepository::delete → KILLED |
articlesRepository.delete(article); |
| 135 |
1
1. deleteArticle : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::deleteArticle → KILLED |
return genericMessage("Article with id %s deleted".formatted(id)); |
| 136 | } | |
| 137 | ||
| 138 | /** | |
| 139 | * Update a single article | |
| 140 | * | |
| 141 | * @param id the id of the article to update | |
| 142 | * @param incoming the new article data | |
| 143 | * @return the updated article | |
| 144 | */ | |
| 145 | @Operation(summary= "Update a single article") | |
| 146 | @PreAuthorize("hasRole('ROLE_ADMIN')") | |
| 147 | @PutMapping("") | |
| 148 | public Article updateArticle( | |
| 149 | @Parameter(name="id") @RequestParam Long id, | |
| 150 | @RequestBody @Valid Article incoming) { | |
| 151 | ||
| 152 | Article article = articlesRepository.findById(id) | |
| 153 |
1
1. lambda$updateArticle$2 : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::lambda$updateArticle$2 → KILLED |
.orElseThrow(() -> new EntityNotFoundException(Article.class, id)); |
| 154 | ||
| 155 |
1
1. updateArticle : removed call to edu/ucsb/cs156/example/entities/Article::setTitle → KILLED |
article.setTitle(incoming.getTitle()); |
| 156 |
1
1. updateArticle : removed call to edu/ucsb/cs156/example/entities/Article::setUrl → KILLED |
article.setUrl(incoming.getUrl()); |
| 157 |
1
1. updateArticle : removed call to edu/ucsb/cs156/example/entities/Article::setExplanation → KILLED |
article.setExplanation(incoming.getExplanation()); |
| 158 |
1
1. updateArticle : removed call to edu/ucsb/cs156/example/entities/Article::setEmail → KILLED |
article.setEmail(incoming.getEmail()); |
| 159 |
1
1. updateArticle : removed call to edu/ucsb/cs156/example/entities/Article::setDateAdded → KILLED |
article.setDateAdded(incoming.getDateAdded()); |
| 160 | ||
| 161 | articlesRepository.save(article); | |
| 162 | ||
| 163 |
1
1. updateArticle : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::updateArticle → KILLED |
return article; |
| 164 | } | |
| 165 | ||
| 166 | /** | |
| 167 | * Get articles within a date range | |
| 168 | * | |
| 169 | * @param startDate the start date in ISO format (yyyy-MM-dd or yyyy-MM-ddTHH:mm:ss) | |
| 170 | * @param endDate the end date in ISO format (yyyy-MM-dd or yyyy-MM-ddTHH:mm:ss) | |
| 171 | * @return a list of Articles within the date range | |
| 172 | */ | |
| 173 | @Operation(summary= "Get articles within a date range") | |
| 174 | @PreAuthorize("hasRole('ROLE_USER')") | |
| 175 | @GetMapping("/bydate") | |
| 176 | public List<Article> getArticlesByDateRange( | |
| 177 | @Parameter(name="startDate") @RequestParam String startDate, | |
| 178 | @Parameter(name="endDate") @RequestParam String endDate) { | |
| 179 | | |
| 180 | LocalDateTime start, end; | |
| 181 | try { | |
| 182 | start = parseDateTime(startDate); | |
| 183 | end = parseDateTime(endDate); | |
| 184 | } catch (DateTimeParseException e) { | |
| 185 | throw new ResponseStatusException(HttpStatus.BAD_REQUEST, "Invalid date format. Use ISO date format (YYYY-MM-DD) or ISO date-time format (YYYY-MM-DDThh:mm:ss)"); | |
| 186 | } | |
| 187 | | |
| 188 |
1
1. getArticlesByDateRange : replaced return value with Collections.emptyList for edu/ucsb/cs156/example/controllers/ArticlesController::getArticlesByDateRange → KILLED |
return articlesRepository.findByDateAddedBetween(start, end); |
| 189 | } | |
| 190 | | |
| 191 | /** | |
| 192 | * Helper method to parse date strings in either date-only or date-time format | |
| 193 | */ | |
| 194 | private LocalDateTime parseDateTime(String dateString) { | |
| 195 |
1
1. parseDateTime : negated conditional → KILLED |
if (dateString.contains("T")) { |
| 196 | // Full date-time format | |
| 197 |
1
1. parseDateTime : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::parseDateTime → KILLED |
return LocalDateTime.parse(dateString); |
| 198 | } else { | |
| 199 | // Date-only format, append T00:00:00 for midnight | |
| 200 |
1
1. parseDateTime : replaced return value with null for edu/ucsb/cs156/example/controllers/ArticlesController::parseDateTime → KILLED |
return LocalDateTime.parse(dateString + "T00:00:00"); |
| 201 | } | |
| 202 | } | |
| 203 | } | |
Mutations | ||
| 57 |
1.1 |
|
| 72 |
1.1 |
|
| 74 |
1.1 |
|
| 117 |
1.1 |
|
| 132 |
1.1 |
|
| 134 |
1.1 |
|
| 135 |
1.1 |
|
| 153 |
1.1 |
|
| 155 |
1.1 |
|
| 156 |
1.1 |
|
| 157 |
1.1 |
|
| 158 |
1.1 |
|
| 159 |
1.1 |
|
| 163 |
1.1 |
|
| 188 |
1.1 |
|
| 195 |
1.1 |
|
| 197 |
1.1 |
|
| 200 |
1.1 |