Skip to content

Latest commit

 

History

History
327 lines (228 loc) · 14.4 KB

README.md

File metadata and controls

327 lines (228 loc) · 14.4 KB

🐿 도토리 마켓

Rest API 통신을 이용한 오픈마켓 앱

1. 프로젝트 소개

1) 진행

  • 개발자 : 릴리, 예거 (팀 프로젝트 + 개인 프로젝트)
  • 1차 : 2022.1. 3 ~ 1. 28 (페어프로그래밍/ MVC 설계로 네트워크 모듈, 상품 등록, 리스트 기능 구현)
  • 2차: 2022. 6. 13 ~ 7. 28 (개인/ MVVM Clean + rxSwift 설계로 리팩터링, 상품 수정, 좋아요 기능 구현)
  • 코드 리뷰 진행 / 리뷰어 : 찰리, 숲재

2) 코드 리뷰

STEP 구현 기능 해당 PR
STEP 1 모델, 네트워크 모듈 설계, 네트워크 테스트 yagom-academy/ios-open-market#89
STEP 2 상품 리스트(Table/CollectionVIew), 캐싱, pagination yagom-academy/ios-open-market#101
STEP 3 상품 상세 화면, ImagePicker, RefreshControl yagom-academy/ios-open-market#121
MVVM 리팩터링 상품 등록, 사용자 인풋 Validation, rxSwift+MVVM으로 구조 리팩터링, 네트워크와 무관한 ViewModel 테스트 #13

3) 구현 기능

상품 보기 (리스트/그리드뷰 지원)

  • UITableViewController, UICollectionViewController 로 상품 정보를 표시
  • 제공하는 데이터가 동일하기 때문에 ProductListSceneViewModel을 공유
  • willDisplayCell활용 pagination 구현

상품 상세정보 보기

  • UICollectionView + UIPageControl로 여러 이미지 보기 구현
  • ❤️를 눌러 관심 상품 저장

새로운 상품 등록

  • UIImagePickerController로 기기 이미지 첨부 가능
  • 입력 정보의 유효성 검증 및 알맞은 알럿 메시지 사용자에게 표시

상품 수정 기능

  • 키보드 등장시 컨텐츠 가림을 KeyboardNotification을 활용하여 사용성 개선

2. 설계 개요

1) 개발 환경

  • 배포 타겟: iOS 13.2
  • UI : UIKit, Storyboard, AutoLayout
  • Network : URLSession, RESTAPI
  • Database : RealmSwfit
  • RxSwift, RxCocoa, RxTest
  • 의존성 관리 툴 : SPM

2) 프레임워크 선택 시 고려사항

RxSwift

  • 목적 : 모델의 변화를 뷰에 자동으로 업데이트하는 리액티브한 구조 구현
  • 선택 이유
    • 다양한 operator를 제공, 데이터 스트림을 컴바인 하기 편리함
    • RxCocoa 활용시 뷰에서 생성되는 이벤트 관리, 데이터 바인딩 편리
    • 비동기 코드를 동기 코드처럼 선언형으로 작성 가능, 가독성 향상
  • 리스크
    • 메모리 누수 가능성 ( weak self,disposeBag사용하여 retain cycle 방지)

RealmSwift

  • 목적 : 좋아요한 상품, 등록 상품 ID 저장을 위한 persistence DB 구현
  • 선택 이유
    • 간단한 API로 데이터의 영구적 저장 가능
    • 안드로이드와 공유 가능, 현업에서 쓰이는 프레임워크라 학습 목적
    • Object 확장성 좋음
  • 리스크
    • 앱의 용량이 커짐
    • 데이터 스레드 간 공유 불가 (스케줄러로 강제)

3) 아키텍처 및 패턴

MVVM + 클린 아키텍처

도입 이유

  • 뷰모델에서 비지니스 로직을 분리하고, 재사용하기 위해 Usecase를 도입
  • 데이터 소스 의존성을 낮추기 위해 Repository인터페이스 정의
  • ProductListUsecase를 프로토콜로 정의
  • 모든 상품, 좋아요한 상품, 등록한 상품 뷰에서 사용할 Usecase를 구체타입으로 구현

  • 비즈니스로직이 포함되지 않은 순수한 뷰
  • API, 프레임워크와 도메인 레이어 의존성 분리 (프레임워크, APIService를 변경시 Repository 구현만 수정하면 되고, DTO를 통해 프레임워크와 의존성 분리 가능)
  • 추상화, 의존성 주입을 통한 각 계층 Unit Test 가능

화면 전환을 담당하는 Coordinator Pattern 적용

도입 이유

뷰 컨트롤러에서 화면 전환을 실행하기 때문에, 뷰 컨트롤러를 뷰로써 재사용하기 어려움 코디네이터 객체에게 화면 전환 역할을 위임하여 뷰 컨트롤러에서 비즈니스 로직 분리를 도모

  • 각 탭마다 코디네이터 생성하여, MainCoordinator의 child로 관리

결과

  • 깡통 뷰로써 재사용 가능
  • 뷰 컨트롤러 간 의존성 제거

뷰 재사용, 느슨한 결합를 위한 시도

  1. 같은 뷰 모델이 사용하는 usecase를 프로토콜로 추상화
  2. 같은 뷰에 대한 Coordinator를 프로토콜로 추상화
  3. FactoryPattern으로 같은 뷰 컨트롤러를 공유하는 뷰 생성

스크린샷 2022-07-28 오후 9 40 11

2) 동일 뷰에 대한 Coordinator를 포로토콜로 추상화

  • 코디네이터는 추상적 인터페이스를 정의 (어떠한 액션이 일어나는지에 대해서만 정의)
protocol ProductListCoordinator: Coordinator {
    
    func rightNavigationItemDidTapped(from: UIViewController)
    func cellDidTapped(of productID: Int) 
    
}
  • 각 뷰(모든 상품/ 좋아요한 상품/ 등록 상품) 의 코디네이터를 구체 타입으로 구현
  • 뷰 컨트롤러에 코디네이터를 주입하여 각 뷰별 화면 전환 수행

3)FactoryPattern으로 동일한 뷰컨을 공유하는 뷰 생성

문제 상황 : 코디네이터에서 뷰 생성시 usecase를 주입해야하기 때문에 코디네이터가 도메인을 알게되는 문제 발생

  • 모든 상품, 좋아요한 상품, 등록한 상품 뷰는 동일한 뷰 컨트롤러를 사용
  • 같은 뷰 컨트롤러를 사용하는 뷰를 열거형으로 정의, 팩토리 패턴으로 생성
ProductListViewFactory 구현
struct ProductListViewFactory {
    
    // MARK: - ProductListView Type
    
    enum ProductListView {
        case allProduct
        case likedProuduct
        case myProduct
    }
    
	func make(
	        viewType: ProductListView,
	        coordinator: ProductListCoordinator) -> ProductTableViewController {
	        switch viewType {
	        case .allProduct:
	            return UIStoryboard.main.instantiateViewController(
	                identifier: "ProductTableViewController", creator:  { coder -> ProductTableViewController in
	                    let viewModel = ProductListSceneViewModel(usecase: AllProductListUsecase())
	                    let vc = ProductTableViewController(
	                        viewModel: viewModel,
	                        coordinator: coordinator,
	                        coder: coder)
	                    return vc!
	                })
	                ... 
	  }

}

결과

  • 하나의 뷰를 공유하는 Scene을 하나의 control point에서 관리 할 수 있음
  • 코디네이터에서 비즈니스로직을 분리하고, 각 뷰 컨트롤러간 결합성을 낮출 수 있음

3. 트러블 슈팅

Testable한 Network Layer는 어떻게 만들까?

문제 상황 : API로 데이터를 받아 사용하는 ViewModel을 테스트하기 위해선 Network 객체도 테스터블하게 만들어야했습니다.

해결 방법

  • URLProtocol을 상속한 MockURLProtocol을 구현했습니다.
  • MockURLProtocol에 서버와 통신을 통해 받은 dataresponse대신 mockResponse, data 를 전달하도록 startLoading() 오버라이딩했습니다.
  • 따라서 실제 통신을 하지 않고도 작동하는 mock APIService구현할 수 있었습니다.

➡️ APIService의 코드 수정 없이, MockURLProtocol로 설정한 URLSessionConfiguration을 주입하여 네트워크와 무관하게 바인딩, 뷰 모델 로직 테스트 가능해졌습니다.

test: MockURLProtocol 구현 및 테스트 코드 적용

Observable 스트림을 갖는 ViewModel Test

문제 상황

  1. 실제 사용자 이벤트 대신 테스트 이벤트를 조작해야합니다.
  2. Input에 들어갈 ControlProperty 타입을 만들어야합니다.
  3. ViewModelUsecase에 의존하기 때문에 Usecase의 동작에 영향을 받습니다.

해결 방법

  • TestSchedulerColdObservable 을 생성하여 사용자 이벤트를 대신할 이벤트를 예약한후, ControlProperty의 값을 나타낼 BehaviorSubject에 바인드했습니다.
  • ControlProperty는 최초의 UIVaule를 이벤트로 내보냅니다. 따라서 BehaviorSubject로 최초 값을 정의하고 실제와 동일한 테스트 환경을 만들었습니다.
  • input에서 output으로 변환되는 과정에서 Usecase를 사용해야하만 하는 경우가 있었습니다. (다수의 Field에서 input을 받아 Usecase에서 검증하는 경우) Usecase만을 먼저 독립적으로 테스트하고(MockRepository 사용) Usecase 의 신뢰성을 검증 한 후 ViewModel을 테스트했습니다.
  • ViewModelUsecase 간 강한 결합이 있다는 문제를 인지했고 뷰 모델을 더 단순화해야겠다고 느꼈습니다.
  • 테스트 코드는 버그의 조기발견, 안정성뿐만 아니라 더 나은 코드를 작성하는데 도움이 된다는 것을 느낄 수 있던 경험이었습니다.

RxSwift 에러 핸들링

문제 상황 : 뷰 컨트롤러에서 각 에러에 따른 알림을 사용자에게 차별적으로 보여주려면 에러를 구독해합니다. 하지만 에러가 한 번 방출되면 스트림은 종료되고, 다시 이벤트를 받지 못합니다.

시도

  • retry를 쓰면 에러가 방출되어도 그 즉시 dispose되고 다시 구독되기 때문에 스트림을 살릴 수 있지만, 에러가 옵저버에게 전달되지 않기 때문에 뷰에서 어떤 에러가 방출됐는지 알 수 없습니다.
  • catch(onErrorJustReturn:), asDriver(onErrorJustReturn:) 를 쓰면 에러대신 다른 값을 넥스트로 보내고 complete되기 때문에 더 이상 사용자 이벤트를 받을 수 없습니다.
  • Result타입으로 값과 에러를 wrapping해서 next로 방출하면 뷰에서 핸들링이 가능합니다. 하지만 뷰컨이 분기를 하는 것과 같은 로직을 갖게되어 역할 분리 측면에선 좋은 방법은 아니라 판단했습니다.

해결 방법

  • 뷰에 AlertController의 뷰모델을 전달하는 스트림을 Subject타입으로 구현했습니다.
  • 스트림에서 error방출시 do에서 Subject에게 AlertViewModel을 next로 보냈습니다.
protocol AlertViewModel {
    
    var title: String? { get }
    var message: String? { get }
    var actionTitle: String? { get }
}

통신 절감 및 빠른 이미지 보기 위한 Cache 구현

문제 상황 : 이미지는 용량이 크기 때문에 스크롤 할 때마다 셀에 느리게 바인딩되는 상황이었습니다.

해결 방법

  • UIImage+ExtensionNScache를 사용해 해당 이미지 URL을 key로 데이터를 캐싱했습니다.

➡️ 사용성과 통신 비용을 개선했습니다.


4. 학습한 내용

Escaping Closure 사용시 강한 순환 참조 해결 방법

Escaping Closure는 참조타입으로 강한 순환 참조로 인한 메모리 누수를 유발할 수 있습니다.

  • 객체의 reference count를 증가 시켜야할 상황인지 판단해 참조를 해야한다는 것, [weak self] unowned로 강한 순환참조를 해결할 수 있다는 점을 학습했습니다.

  • RxSwift의 operator들은 escaping closure로 정의 되어있습니다. 따라서 뷰컨트롤러를 [weak self]로 참조해야만 disposeBag이 작동될 때 모든 스트림이 종료되고, 뷰 컨트롤러도 해제될 수 있습니다.

HTTP구조와 URLSession을 사용한 통신

  • 직접 HTTP header, body, httpMethod, boundary를 구현해보며 HTTP 프로토콜의 구조에 대해 학습할 수 있었습니다.
  • APIRequest를 프로토콜로 정의하여, 개별 Request 생성시 프로토콜을 구현했습니다.
  • POP를 직접 적용해봄으로싸 확장성과 유연성을 이해할 수 있었습니다.
protocol APIRequest {
   // Request에 대한 Response Parsing Type을 연관 타입으로 정의
    associatedtype Response: ResponseDataType
    
    var url: URL? { get }
    var httpMethod: HTTPMethod { get }
    var header: [String: String] { get }
    var body: Data? { get }
    
}

extension APIRequest {
    
    // 프로토콜 기본 구현을 활용해 편리하게 URLRequest를 생성하는 API 제공
    func urlRequest() -> URLRequest? {
        guard let url = url else {
            return nil
        }
        
        var urlRequest = URLRequest(url: url)
        urlRequest.httpMethod = httpMethod.rawValue
        urlRequest.allHTTPHeaderFields = header
        urlRequest.httpBody = body
        return urlRequest
    }
    
}