Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat/#127 ManiaDB API 를 사용하여 등록되지 않은 노래 검색 기능 구현 #153

Open
wants to merge 13 commits into
base: main
Choose a base branch
from
9 changes: 9 additions & 0 deletions backend/build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,12 @@ dependencies {
implementation 'org.springframework.boot:spring-boot-starter-web'
implementation 'org.springframework.boot:spring-boot-starter-validation'

// WebClient Dependency
implementation 'org.springframework.boot:spring-boot-starter-webflux'

// XML parsing Dependency
implementation 'org.glassfish.jaxb:jaxb-runtime'

compileOnly 'org.projectlombok:lombok'

runtimeOnly 'com.h2database:h2'
Expand All @@ -36,6 +42,9 @@ dependencies {

testImplementation 'org.springframework.boot:spring-boot-starter-test'
testImplementation 'io.rest-assured:rest-assured:5.3.1'

// WebClient Test Dependencies
testImplementation 'com.squareup.okhttp3:mockwebserver'
}

tasks.named('test') {
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,70 @@
package shook.shook.song.application;

import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.List;
import java.util.Objects;
import lombok.RequiredArgsConstructor;
import org.springframework.http.HttpStatusCode;
import org.springframework.http.MediaType;
import org.springframework.stereotype.Service;
import org.springframework.web.reactive.function.client.WebClient;
import shook.shook.song.application.dto.UnregisteredSongSearchResponse;
import shook.shook.song.application.dto.maniadb.ManiaDBAPISearchResponse;
import shook.shook.song.application.dto.maniadb.UnregisteredSongResponses;
import shook.shook.song.exception.UnregisteredSongException;
import shook.shook.song.exception.UnregisteredSongException.EmptyResultException;

@RequiredArgsConstructor
@Service
public class ManiaDBSearchService {

private static final String MANIA_DB_API_URI = "/%s/?sr=song&display=%d&key=example&v=0.5";
private static final int SEARCH_SIZE = 100;
private static final String SPECIAL_MARK_REGEX = "[^ㄱ-ㅎㅏ-ㅣ가-힣a-zA-Z0-9,. ]";
private final WebClient webClient;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

상수와 필드 사이에 줄 간격 두면 가독성이 더 좋아질 것 같아요!


public List<UnregisteredSongSearchResponse> searchSongs(final String searchWord) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

인자로 들어오는 searchWord가 비어있는 문자열일 때에 대한 검증이 필요할 것 같아요!! 😁

image

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

헉 꼼꼼한 예외 처리 👍
검색어가 null 일 때, 빈 문자열일 때 예외를 처리하는 로직을 추가했습니다.

final String parsedSearchWord = replaceSpecialMark(searchWord);
final UnregisteredSongResponses searchResult = getSearchResult(parsedSearchWord);

if (Objects.isNull(searchResult.getSongs())) {
return Collections.emptyList();
}

return searchResult.getSongs().stream()
.map(UnregisteredSongSearchResponse::from)
.toList();
}

private String replaceSpecialMark(final String rawSearchWord) {
return rawSearchWord.replaceAll(SPECIAL_MARK_REGEX, "");
}

private UnregisteredSongResponses getSearchResult(final String searchWord) {
final String searchUrl = String.format(MANIA_DB_API_URI, searchWord, SEARCH_SIZE);
final ManiaDBAPISearchResponse result = getResultFromManiaDB(searchUrl);

if (Objects.isNull(result)) {
throw new EmptyResultException();
}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

제가 생각한 해당 분기에 들어오는 경우입니다.

  1. getResultFromManiaDB 메서드를 실행한다.
  2. webClient 통신에 성공한다.
  3. 응답코드가 4xx, 5xx 가 아니다.
  4. 응답 본문을 ManiaDBAPISearchResponse.class 로 역직렬화하는 과정에서 예외가 발생하지 않는다.
  5. 역직렬화된 maniaDBAPISearchResponse 가 null 이다.

언제 해당 상황이 발생하고 이 때 던지는 예외가 EmptyResultException 인 이유도 궁금해요~😊

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💬 저도 스플릿이랑 비슷한 질문입니다 ㅎ.ㅎ

처음에 서비스 코드만 봤을 때는 EmptyResultException 이어서 "검색 결과가 없으면 예외가 발생하는건가~?" 라고 생각했는데, 테스트 코드를 보니 maniaDB에서 EMPTY_STRING ("") 이 반환될 때에 대한 예외 처리인 것 같더라구요!

Response 객체가 null이 되는 상황이 maniaDB에서 EMPTY_STRING을 반환할 때만인지, 아니면 다른 상황도 있는지 궁금해요!

그리구 빈 문자열만 해당한다면 언제 이런 상황이 생기는지도 궁금합니다

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

앗 그리구 UnregisteredSongException.EmptyResultException() 으로 통일해야 할 것 같아요~!

Copy link
Collaborator Author

@Cyma-s Cyma-s Jul 31, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

스플릿이 말한 플로우가 맞습니다.

아직 저 예외를 만나본 적은 없어요.
일단 getSongs()getSingers() 같은 값들에 null 처리를 해준 이유는 응답에 DTO의 @XmlRootElement에 해당하는 XML 태그가 존재하지 않은 경우에 null이 들어가기 때문입니다.

final ManiaDBAPISearchResponse result = getResultFromManiaDB(searchUrl); 부분은 ManiaDB에서 응답 값을 ManiaDBAPISearchResponse 로 변경하고 있습니다.
이때, ManiaDB에서 응답하는 값에 rss 태그가 존재하지 않으면 result 값이 null 이 될 가능성이 있습니다.

예외를 경험한 것은 아니지만, 충분히 예측 가능한 예외라고 생각하여 null 체크를 해서 예외를 던지도록 했습니다. 좀 더 생각해보니 이 경우는 ManiaDBServerException 이라고 볼 수도 있겠네요 🤔

++ 상위 예외가 자동으로 import 됐나 보네요 ㅠㅠ 한 3번쯤 확인한 것 같은데 어째서!


return result.getSongs();
}

private ManiaDBAPISearchResponse getResultFromManiaDB(final String searchUrl) {
return webClient.get()
.uri(searchUrl)
.accept(MediaType.TEXT_XML)
.acceptCharset(StandardCharsets.UTF_8)
.retrieve()
.onStatus(HttpStatusCode::is4xxClientError, (clientResponse) -> {
throw new UnregisteredSongException.ManiaDBClientException();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

개인적으로 등록되지 않은 노래 예외라는 뜻의 UnregisteredSongException 의 inner class로 ManiaDBClientException 이 있는 것이 조금 부자연스러워 보이는 것 같아요~

Suggested change
throw new UnregisteredSongException.ManiaDBClientException();
throw new ExternalApiException.ManiaDBClientException();

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

좋은 피드백 감사합니다! 사실 이름 짓기가 너무 어려워서 ㅋㅋㅋ ExternalApiException 좋네요 👍

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

예외처리 기가막히고 신기하네요 👍

})
.onStatus(HttpStatusCode::is5xxServerError, (clientResponse) -> {
throw new UnregisteredSongException.ManiaDBServerException();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

위에 코멘트와 동일합니다!!�

Suggested change
throw new UnregisteredSongException.ManiaDBServerException();
throw new ExternalApiException.ManiaDBServerException();

})
.bodyToMono(ManiaDBAPISearchResponse.class)
.block();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
package shook.shook.song.application.dto;

import java.util.stream.Collectors;
import lombok.AllArgsConstructor;
import lombok.Getter;
import shook.shook.song.application.dto.maniadb.SongArtistResponse;

@AllArgsConstructor
@Getter
public class UnregisteredSongSearchResponse {

private static final String EMPTY_SINGER = "";
private static final String SINGER_DELIMITER = ", ";
private String title;
private String singer;
private String albumImageUrl;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

여기도 위와 같은 리뷰입니다!


public static UnregisteredSongSearchResponse from(
final shook.shook.song.application.dto.maniadb.UnregisteredSongResponse unregisteredSongResponse) {
if (unregisteredSongResponse.getTrackArtists() == null
|| unregisteredSongResponse.getTrackArtists().getArtists() == null) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. final 뒤의 UnregisteredSongResponse의 경로는 import문으로 뺄 수 있을 것 같습니다.
  2. if문에서 가수 이름이 null인지 아닌지를 파악하는 것 같습니다. 먼저 unregisteredSongResponse의 SongtrackArtistsResponse(trackArtiests)객체가 null인지 파악하고 그 내부의 리스트가 null인지 파악하는데 혹시 데이터를 받아올 때 객체 내부의 값이null로 들어오는 경우가 있어서 나누어 놓으신건가요?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. 코드 작성하다가 놓친 것 같네용 수정했습니다!
  2. 넹 이해하신 내용이 맞습니다 😊 위 코멘트에 설명해놓았듯이 XML 태그가 없는 경우 필드에 null 값이 들어가게 됩니다. track_artist tag, artist tag 가 존재하지 않을 때는 모두 가수 이름이 없는 것으로 간주하고 EMPTY_STRING을 리턴하도록 했습니다 👍🏻

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💬 요 if문 내의 조건이 어떤걸 의미하는지 잘 나타낼 수 있게 따로 메서드로 분리하는건 어떻게 생각하시나요 ?.?
저는 처음 코드를 봤을 때 의미가 한 번에 이해되지 않았어서 남겨봅니다 ㅎ.ㅎ

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

좋은 피드백이네요!!! 😄
처음 코드를 보는 사람은 확실히 이해하기 힘든 부분인 것 같아요. 메서드 분리했습니다 ˙ᵕ˙

return new UnregisteredSongSearchResponse(
unregisteredSongResponse.getTitle().trim(),
EMPTY_SINGER,
unregisteredSongResponse.getAlbum().getImage().trim()
);
}

final String singers = collectToString(unregisteredSongResponse);

return new UnregisteredSongSearchResponse(
unregisteredSongResponse.getTitle().trim(),
singers,
unregisteredSongResponse.getAlbum().getImage().trim()
);
}
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

현재 정적 팩토리 메소드를 이용하고 있는데 @AllArgsConstructor에 따로 접근제어를 하지 않았습니다. public으로 한 것이 의도한게 아니라면 private로 접근제어자를 추가하면 좋을 것 같습니다.


private static String collectToString(
final shook.shook.song.application.dto.maniadb.UnregisteredSongResponse unregisteredSongResponse) {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

여기도 import문으로 경로를 나타낼 수 있을 것 같습니다.

return unregisteredSongResponse.getTrackArtists().getArtists().stream()
.map(SongArtistResponse::getName)
.map(String::trim)
.collect(Collectors.joining(SINGER_DELIMITER));
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package shook.shook.song.application.dto.maniadb;

import jakarta.xml.bind.annotation.XmlElement;
import jakarta.xml.bind.annotation.XmlRootElement;
import lombok.Getter;

@Getter
@XmlRootElement(name = "rss")
public class ManiaDBAPISearchResponse {

@XmlElement(name = "channel")
private UnregisteredSongResponses songs;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package shook.shook.song.application.dto.maniadb;

import jakarta.xml.bind.annotation.XmlElement;
import jakarta.xml.bind.annotation.XmlRootElement;
import lombok.Getter;

@Getter
@XmlRootElement(name = "album", namespace = "http://www.maniadb.com/api")
public class SongAlbumResponse {

@XmlElement(name = "image")
private String image;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
package shook.shook.song.application.dto.maniadb;

import jakarta.xml.bind.annotation.XmlElement;
import jakarta.xml.bind.annotation.XmlRootElement;
import lombok.Getter;

@Getter
@XmlRootElement(name = "artist", namespace = "http://www.maniadb.com/api")
public class SongArtistResponse {

@XmlElement(name = "name")
private String name;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
package shook.shook.song.application.dto.maniadb;

import jakarta.xml.bind.annotation.XmlElement;
import jakarta.xml.bind.annotation.XmlRootElement;
import java.util.List;
import lombok.Getter;

@Getter
@XmlRootElement(name = "trackartists", namespace = "http://www.maniadb.com/api")
public class SongTrackArtistsResponse {

@XmlElement(name = "artist", namespace = "http://www.maniadb.com/api")
private List<SongArtistResponse> artists;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
package shook.shook.song.application.dto.maniadb;

import jakarta.xml.bind.annotation.XmlElement;
import jakarta.xml.bind.annotation.XmlRootElement;
import lombok.Getter;

@Getter
@XmlRootElement(name = "item")
public class UnregisteredSongResponse {

@XmlElement(name = "title")
private String title;

@XmlElement(name = "trackartists", namespace = "http://www.maniadb.com/api")
private SongTrackArtistsResponse trackArtists;

@XmlElement(name = "album", namespace = "http://www.maniadb.com/api")
private SongAlbumResponse album;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
package shook.shook.song.application.dto.maniadb;

import jakarta.xml.bind.annotation.XmlElement;
import jakarta.xml.bind.annotation.XmlRootElement;
import java.util.List;
import lombok.Getter;

@Getter
@XmlRootElement(name = "channel")
public class UnregisteredSongResponses {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

등록되지 않은 노래 목록 응답이라는 뜻 UnregisteredSongResponses 네이밍을 사용했네요~

개인적으로 아직 ManiaDB가 DB에 없는 노래만 검색하기 위해서 사용한다는 정책이 모두와 같이 결정된 사항이 아니기도 하고

ManiaDBSearchService.searchSongs() 메서드의 반환 객체 네이밍이 UnregisteredSongResponses 인게 저는 부자연스러운 것 같다고 느껴져요~

어차피 ManiaDB api에 종속적인 코드들에서 사용되는 DTO기에 개인적으로는 아래 네이밍이 어떨까 싶어요~

Suggested change
public class UnregisteredSongResponses {
public class SearchedSongsWithManidaDbApiResponses {

혹시 추후에 등록되지 않은 노래의 검색, 등록된 노래의 검색 을 나누었을 때 등록되지 않는 노래의 검색을 mainaDB api가 아닌 다른 방법으로 구현할 수도 있다는 확장성을 위해 택한 네이밍일까요??

Copy link
Collaborator Author

@Cyma-s Cyma-s Jul 31, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

넹 이해하신 내용이 맞습니다
등록되지 않은 노래의 검색 / 등록된 노래의 검색이 분리되어 있을 것이라고 가정하고 DTO 이름을 설정해서 UnregisteredSongResponse 라고 하긴 했어요!

그렇다면 ManiaDB에서 가져오는 결과라는 뜻을 좀 더 확실하게 담기 위해 SearchedSongFromManiaDBApiResponse 라는 네이밍은 어떤가요?
이 부분은 스플릿 코멘트 확인 후에 반영하겠습니다~ 😄

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

i like it👍


@XmlElement(name = "item")
private List<UnregisteredSongResponse> songs;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
package shook.shook.song.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.codec.xml.Jaxb2XmlDecoder;
import org.springframework.web.reactive.function.client.ExchangeStrategies;
import org.springframework.web.reactive.function.client.WebClient;

@Configuration
public class ManiaDBConfiguration {

private static final String MANIA_DB_BASE_URL = "http://www.maniadb.com/api/search";

@Bean
public WebClient getWebClient() {
return WebClient.builder()
.baseUrl(MANIA_DB_BASE_URL)
.exchangeStrategies(
ExchangeStrategies.builder()
.codecs(configurer ->
configurer.defaultCodecs().jaxb2Decoder(new Jaxb2XmlDecoder())
)
.codecs(configurer ->
configurer.defaultCodecs().maxInMemorySize(4 * 1024 * 1024)
)
.build()
)
.build();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,25 @@
package shook.shook.song.exception;

public class UnregisteredSongException extends RuntimeException {

public static class EmptyResultException extends UnregisteredSongException {

public EmptyResultException() {
super();
}
}

public static class ManiaDBServerException extends UnregisteredSongException {

public ManiaDBServerException() {
super();
}
}

public static class ManiaDBClientException extends UnregisteredSongException {

public ManiaDBClientException() {
super();
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
package shook.shook.song.ui;

import java.util.List;
import lombok.RequiredArgsConstructor;
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import shook.shook.song.application.ManiaDBSearchService;
import shook.shook.song.application.dto.UnregisteredSongSearchResponse;

@RequiredArgsConstructor
@RequestMapping("/songs/unregistered/search")
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

등록되지 않은 상황에서만으로 가정한 기능이였군요!

위에 코멘트에 대한 의문이 풀렸습니다~ 👍

@RestController
public class ManiaDBApiController {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

개인적으로 "/songs/unregistered/search" 에 대한 컨트롤러 이름이 ManiaDBApiController인 것이 조금 부자연스러운 것 같아요!

UnregisteredSongSearchController 인터페이스, UnregisteredSongSearchService 인터페이스 분리를 통하여 ManiaDB를 사용하는 구현체를 둘 수 있는 방법에 대해서 베로도 알고 있을 거에요.

아마 베로도 인터페이스로 모두 분리하기에는 굳이? 와 오버엔지니어링일 꺼라고 생각했을 것 같고 저도 그렇게 생각합니다!


Controller 이름만을 UnregisteredSongSearchController로 아예 바꾸고 내부에서 사용하는 Service만 ManiaDBSearchService 인 것은 어떨까요??

컨트롤러 이름이 ManiaDBApiController 지만 사실상 하는 일은 searchUnregisteredSong() 이고 컨트롤러의 네임과 역할이 맞지 않는 것 같다는 생각이 들어요~

구체적인 비즈니스 로직을 수행하는 Service 레이어의 타입에만 ManiaDB이라는 단어를 포함해도 충분하지 않을 까라는 생각을 해봅니다~

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

오 좋은 피드백 감사합니다 👍
확실히 구체적인 비즈니스 로직을 수행하는 Service 이름이 구체적인 건 좋지만 Controller 이름이 API 이름으로 되어 있는 건 좀 부자연스럽기도 하네요 🤔


private final ManiaDBSearchService maniaDBSearchService;

@GetMapping
public ResponseEntity<List<UnregisteredSongSearchResponse>> searchUnregisteredSong(
final @RequestParam("keyword") String searchWord
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

베로쨩~
맨 밑에 첨부한 이미지처럼
localhost:8080/songs/unregistered/search?keyword=

keyword에 빈 값을 넣어서 보냈을 때,
Connection refused: no further information 500 에러가 발생하더라구요!

여기에 대한 예외 처리와 테스트 코드도 있으면 좋을 것 같습니다 ㅎ.ㅎ

image

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

이 부분 빠졌었군요 😅 역시 코드 리뷰의 중요성...
빈 값 테스트 코드 추가했습니다!

) {
final List<UnregisteredSongSearchResponse> songs = maniaDBSearchService.searchSongs(
searchWord);

return ResponseEntity.ok(songs);
}
}
Loading