Skip to content

Latest commit

 

History

History
251 lines (206 loc) · 10.7 KB

24.02.05 유튜브 API 사용하기.md

File metadata and controls

251 lines (206 loc) · 10.7 KB

https://console.cloud.google.com/welcome/new?supportedpurview=project 구글 클라우드 서비스에서 프로젝트를 생성하고 유튜브 API3 키 를 받으면 유튜브 API를 쓸 수 있다.

Interface

유튜브 api 문서는 https://developers.google.com/youtube/v3?hl=ko 이곳에서 참조 탭을 누르면 확인 할 수 있다.

검색에서 리스트를 누르고

문서를 보면

이런식으로 데이터를 요청할때 넣어야 할 값들을 확인 할 수 있다.

요청 인터페이스를 만들었는데

interface YoutubeSearchService {

    @GET("search")
    suspend fun searchResult (
        @Query("forContentOwner") forContentOwner: String? = null, // 콘텐츠 소유자 동영상 검색 아래는 내 동영상
        @Query("forMine") forMine: String? = null, // 위랑 아래랑 둘중하나 혹은 0
        @Query("channelId") channelId: String? = null, //
        @Query("eventType") eventType: String? = null, // completed - 완료된 브로드캐스트, live , upcoming - 예정된 방송
        @Query("maxResults") maxResults: Int? = 50,
        @Query("part") part: String = "snippet",
        @Query("order") order: String = "relevance", //date – 리소스를 만든 날짜를 기준, rating – 높은 평가부터 낮은 평가순, relevance – 검색어와의 관련성을 기준, title – 제목알파벳순, viewCount – 리소스가 조회수가 높은 순
        @Query("regionCode") regionCode: String = "kr", // 국가에서 볼 수 있는 동영상
        @Query("pageToken") pageToken: String? = null,
        @Query("relevanceLanguage") relevanceLanguage: String = "ko", // 언어랑 관련성 높은 영상
        @Query("q") query: String = "react", // 검색어
        @Query("videoDuration") videoDuration: String = "any", // long – 20분보다 긴 동영상, medium – 4분에서 20분 사이의 동영상, short – 4분 미만의 동영상
        @Query("type") type: String = "video", // channel, playlist, video
        @Query("videoType") videoType: String = "any", // any – 모든 동영상을 반환합니다, episode - 프로그램의 에피소드만 검색합니다., movie - 영화만 검색합니다
        @Query("videoSyndicated") videoSyndicated: String = "true", // any – 배급 여부에 관계 없이 모든 동영상을 반환합니다.,true – 배급된 동영상만 검색합니다. 외부에서 재생할 수 있는 동영상
    ) : SearchApiResponse

이렇게 서버에 다양한 방식으로 요청 할 수 있다.

이제 데이터를 받을 모델을 만들자

DTO

문서의 오른쪽을 보면 api호출되는 데이터를 실험해 볼 수 있다.

여기에 값을 입력하면 json형식으로 서버에서 데이터를 보내게 되는데

http://json.parser.online.fr/ 를 이용하면 편하게 구조를 볼 수 있다.

여기서 보여주는 kind etag등에 맞춰서 모델을 만들어 보았다.

data class GenericApiResponse<T>(
    val etag: String?,
    val items: List<T>?,
    val kind: String?,
    val regionCode: String?,
    val nextPageToken: String?,
    val prevPageToken: String?,
    val pageInfo: PageInfo?
)
data class PageInfo(
    val resultsPerPage: Int?,
    val totalResults: Int?
)

리스폰 하는 데이터가 굉장히 많았는데

대부분의 정보를 하나로 받아둘 엔티티를 만들어 줘야 한다

data class VideoEntity(
    val id: String?,
    val publishedAt: String?,
    val channelId: String?,
    val title: String?,
    val description: String?,
    val thumbnailHigh: String?,
    val thumbnailMedium: String?,
    val thumbnailLow: String?,
    val channelTitle: String?,
    val tags: List<String>?,
    val categoryId: String?,
    val liveBroadcastContent: String?,
    val defaultLanguage: String?,
    val localizedTitle: String?,
    val localizedDescription: String?,
    val defaultAudioLanguage: String?,
    val duration: String?,
    val dimension: String?,
    val definition: String?,
    val caption: String?,
    val licensedContent: Boolean?,
    val projection: String?,
    val viewCount: String?,
    val likeCount: String?,
    val favoriteCount: String?,
    val commentCount: String?,
    val player: String?,
    val topicDetails: List<String>?,
)

이제 레포지토리를 만들어서 이 모델에 맞게 데이터를 변경 시켜줘야한다.

class ApiRepositoryImpl @Inject constructor(
    private val remoteDataSource: YoutubeSearchService,
) : ApiRepository {
    ...
    override suspend fun getPopularVideo()
            : List<VideoEntity>? = remoteDataSource.getPopularVideo().let {
        nextPageToken = it.nextPageToken
        it.items?.map { item ->
            convertVideoEntity(item)
        }
    }
    
    private fun convertVideoEntity(item: ItemResponse): VideoEntity {
        return VideoEntity(
            id = item.id ?: "",
            publishedAt = item.snippet?.publishedAt ?: "",
            channelId = item.snippet?.channelId ?: "",
            title = item.snippet?.title ?: "",
            description = item.snippet?.description ?: "",
            thumbnailHigh = item.snippet?.thumbnails?.high?.url ?: "",
            thumbnailMedium = item.snippet?.thumbnails?.medium?.url ?: "",
            thumbnailLow = item.snippet?.thumbnails?.default?.url ?: "",
            channelTitle = item.snippet?.channelTitle ?: "",
            tags = item.snippet?.tags?.toList(),
            categoryId = item.snippet?.categoryId ?: "",
            liveBroadcastContent = item.snippet?.liveBroadcastContent ?: "",
            defaultLanguage = item.snippet?.defaultLanguage ?: "",
            localizedTitle = item.snippet?.localized?.title ?: "",
            localizedDescription = item.snippet?.localized?.description ?: "",
            defaultAudioLanguage = item.snippet?.defaultAudioLanguage ?: "",
            duration = item.contentDetails?.duration ?: "",
            dimension = item.contentDetails?.dimension ?: "",
            definition = item.contentDetails?.definition ?: "",
            caption = item.contentDetails?.caption ?: "",
            licensedContent = item.contentDetails?.licensedContent,
            projection = item.contentDetails?.projection ?: "",
            viewCount = item.statistics?.viewCount ?: "",
            likeCount = item.statistics?.likeCount ?: "",
            favoriteCount = item.statistics?.favoriteCount ?: "",
            commentCount = item.statistics?.commentCount ?: "",
            player = item.player?.embedHtml ?: "",
            topicDetails = item.topicDetails?.topicCategories?.toList()
        )
    }

이렇게 하면 서버의 데이터를 사용하기 쉽게 변환 시키며 데이터를 사용할 수 있다.

사용 설명서

이제 팀원들이 보고 사용할 수 있게 설명서를 작성해줬다.

레포지토리의 함수의 기능을 설명하고 받아오는 데이터 모델의 정보를 적어서

각자의 뷰에 맞게 필요한 데이터만을 들고 갈 수 있게 했다.

interface ApiRepository {
    suspend fun searchResult(query: String): List<SearchEntity>?
    suspend fun getPopularVideo(): List<VideoEntity>?
    suspend fun getContentDetails(idList: List<String>): List<VideoEntity>?
    suspend fun getChannelDetails(idList: List<String>): List<ChannelEntity>?
    suspend fun getComments(videoId: String): List<CommentEntity>?
}

searchResult 검색어를 입력하면 검색 결과를 불러옴

getPopularVideo 호출시 동영상 인기순으로 불러옴

getContentDetails 검색이든 인기동영상이든 불러온 데이터의 비디오 아이디를 입력하면 상세한 정보가지고옴

getChannelDetails 채널의 아이디를 입력하면 채널의 상세한 정보를 가지고옴

getComments 비디오 아이디 입력 시 영상의 댓글을 가지고옴

SearchEntity.kt

id String 영상의 id
publishedAt String 게시일
channelId String 올린 채널 아이디
title String 영상의 제목
description String 영상 상세설명
thumbnailHigh String 큰 사이즈썸네일
thumbnailMedium String 중간 사이즈 썸네일
thumbnailLow String 작은 사이즈 썸네일
channelTitle String 채널명
liveBroadcastContent String 생방송 중인지 표시

VideoEntity.kt

id String 영상의 id
publishedAt String 게시일
channelId String 올린 채널 아이디
title String 영상의 제목
description String 영상 상세설명
thumbnailHigh String 큰 사이즈썸네일
thumbnailMedium String 중간 사이즈 썸네일
thumbnailLow String 작은 사이즈 썸네일
channelTitle String 채널명
tags List 영상 태그
categoryId String 카테고리 id
liveBroadcastContent String 생방송 중인지 표시
defaultLanguage String 영상의 기본 언어
localizedTitle String 로컬라이즈된 제목
localizedDescription String 로컬라이즈된 설명
defaultAudioLanguage String 기본 음성 언어
duration String 영상 길이
dimension String 비디오의 차원, 2d, 3d
definition String 비디오 화질, “hd”고화질, “sd”표준화질
caption String 자막이 사용 가능 여부
licensedContent Boolean 라이선스된 콘텐츠인지 확인
projection String 비디오의 투영 방식, 일반적인 비디오는 "rectangular" , 360도 비디오는 "360" 값을 가짐
viewCount String 조회 수
likeCount String 좋아요 수
favoriteCount String 사용자에 의해 "즐겨찾기"로 추가된 횟수,일반적으로 외부에 공개 안돼서 0만 뜸
commentCount String 댓글 수
player String 동영상 플레이어 Url
topicDetails List 비디오와 연관된 주제의 id 목록, id는 키 값이니까 주의