Layers without DI

Horus-iOS·2023년 8월 11일
0

SceneDelegate

func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) {
    guard let windowScene = (scene as? UIWindowScene) else { return }
    window = UIWindow(windowScene: windowScene)
        
    let navigationController = UINavigationController()
        
    window?.rootViewController = navigationController

    let viewController = AsyncAwaitViewController()
        
    navigationController.pushViewController(viewController, animated: true)
        
    window?.makeKeyAndVisible()
}

Layers

import UIKit
import Combine

// MARK: - Data
struct OMGResponseDTO: Decodable {
    let member: [OMGMemberResponseDTO]
}

extension OMGResponseDTO {
    func toDomain() -> OMG {
        return .init(member: member.compactMap { $0.toDomain() })
    }
}

struct OMGMemberResponseDTO: Decodable {
    let activityName: String
    let name: String
    let mbti: String
    
    enum CodingKeys: String, CodingKey {
        case activityName = "activityname"
        case name
        case mbti
    }
}

extension OMGMemberResponseDTO {
    func toDomain() -> OMGMember {
        return .init(activityName: activityName, name: name, mbti: mbti)
    }
}

import Foundation

protocol Network {
    func request(with url: URL) async throws -> Data
    func request(with url: URL, completion: @escaping (Result<Data, Error>) -> Void)
    func request(with url: URL) -> AnyPublisher<Data, Error>
}

final class DefaultNetwork: Network {
    
    init() {
        
    }
    
    func request(with url: URL) async throws -> Data {
        do {
            let requestResult = try await URLSession.shared.data(from: url)
            guard let httpURLResponse = requestResult.1 as? HTTPURLResponse else {
                throw NetworkError.responseError
            }
            switch httpURLResponse.statusCode {
            case 200:
                return requestResult.0
                
            case 300...500:
                throw NetworkError.statusCode
                
            default:
                throw NetworkError.unknownStatus
                
            }
        } catch {
            throw error
        }
    }
    
    func request(with url: URL, completion: @escaping (Result<Data, Error>) -> Void) {
        let urlRequest = URLRequest(url: url)
        let task = URLSession.shared.dataTask(with: urlRequest, completionHandler: { data, response, error in
            if let error = error {
                DispatchQueue.main.async {
                    completion(.failure(error))
                }
            } else {
                guard let httpURLResponse = response as? HTTPURLResponse else {
                    DispatchQueue.main.async {
                        completion(.failure(NetworkError.responseError))
                    }
                    return
                }
                switch httpURLResponse.statusCode {
                case 200:
                    guard let data = data else {
                        DispatchQueue.main.async {
                            completion(.failure(NetworkError.data))
                        }
                        return
                    }
                    DispatchQueue.main.async {
                        completion(.success(data))
                    }
                    
                case 300...500:
                    DispatchQueue.main.async {
                        completion(.failure(NetworkError.statusCode))
                    }
                    
                default:
                    DispatchQueue.main.async {
                        completion(.failure(NetworkError.unknownStatus))
                    }
                    
                }
            }
        })
        task.resume()
    }
    
    private func request(with url: URL) -> URLSession.DataTaskPublisher {
        let request = URLRequest(url: url)
        return URLSession.shared.dataTaskPublisher(for: request)
    }
    
    func request(with url: URL) -> AnyPublisher<Data, Error> {
        return request(with: url)
            .tryMap { data, response in
                guard let httpResponse = response as? HTTPURLResponse, httpResponse.statusCode == 200 else {
                    throw URLError(.notConnectedToInternet)
                }
                return data
            }
            .eraseToAnyPublisher()
    }
    
}

import Foundation
import Combine

protocol DataTransferService {
    func request(with url: URL) async throws -> OMGResponseDTO
    func request(with url: URL, completion: @escaping (Result<OMGResponseDTO, Error>) -> Void)
    func request(with url: URL) -> AnyPublisher<OMGResponseDTO, Error>
}

enum NetworkError: Error {
    case responseError
    case data
    case statusCode
    case unknownStatus
}

enum DataTransferError: Error {
    case decodeFailure
}

final class DefaultDataTransferService: DataTransferService {
    
    private let network: Network
    private let decoder: JSONDecoder
    
    init() {
        self.network = DefaultNetwork()
        self.decoder = JSONDecoder()
    }
    
    func request(with url: URL) async throws -> OMGResponseDTO {
        do {
            let data = try await network.request(with: url)
            let responseDTO = try decoder.decode(OMGResponseDTO.self, from: data)
            return responseDTO
        } catch {
            throw error
        }
    }
    
    func request(with url: URL, completion: @escaping (Result<OMGResponseDTO, Error>) -> Void) {
        network.request(with: url) { result in
            switch result {
            case .success(let data):
                do {
                    let decoded = try self.decoder.decode(OMGResponseDTO.self, from: data)
                    completion(.success(decoded))
                } catch let error {
                    completion(.failure(error))
                }
                
            case .failure(let error):
                completion(.failure(error))
                
            }
        }
    }
    
    func request(with url: URL) -> AnyPublisher<OMGResponseDTO, Error> {
        return network.request(with: url)
            .decode(type: OMGResponseDTO.self, decoder: decoder)
            .eraseToAnyPublisher()
    }
    
}

// MARK: - Repository
import Foundation

struct OMG {
    let member: [OMGMember]
}

struct OMGMember {
    let activityName: String
    let name: String
    let mbti: String
}

import Foundation
import Combine

protocol AsyncAwaitRepository {
    func request() async throws -> OMG
    func request(completion: @escaping (Result<OMG, Error>) -> Void)
    func request() -> AnyPublisher<OMG, Error>
}

final class DefaultAsyncAwaitRepository: AsyncAwaitRepository {
    
    private let dataTransferService: DataTransferService
    private let url: URL
    
    init() {
        self.dataTransferService = DefaultDataTransferService()
        self.url = URL(string: "") ?? .init(filePath: "")
    }
    
    func request() async throws -> OMG {
        do {
            let responseDTO = try await dataTransferService.request(with: url)
            let domainEntity = responseDTO.toDomain()
            return domainEntity
        } catch let error {
            throw error
        }
    }
    
    func request(completion: @escaping (Result<OMG, Error>) -> Void) {
        dataTransferService.request(with: url) { result in
            switch result {
            case .success(let data):
                let domainEntity = data.toDomain()
                completion(.success(domainEntity))
                
            case .failure(let error):
                completion(.failure(error))
            }
        }
    }
    
    func request() -> AnyPublisher<OMG, Error> {
        dataTransferService.request(with: url)
            .tryMap { data in
                return data.toDomain()
            }
            .mapError({ error in
                return error
            })
            .eraseToAnyPublisher()
    }
    
}

import Foundation
import Combine

// MARK: - Domain
protocol AsyncAwaitUseCase {
    func executeRequest() async throws -> OMG
    func executeRequest(completion: @escaping (Result<OMG, Error>) -> Void)
    func executeRequest() -> AnyPublisher<OMG, Error>
}

final class DefaultAsyncAwaitUseCase: AsyncAwaitUseCase {
    
    private let repository: AsyncAwaitRepository
    
    init() {
        self.repository = DefaultAsyncAwaitRepository()
    }
    
    func executeRequest() async throws -> OMG {
        do {
            let domainEntity = try await repository.request()
            return domainEntity
        } catch {
            throw error
        }
    }
    
    func executeRequest(completion: @escaping (Result<OMG, Error>) -> Void) {
        repository.request(completion: completion)
    }
    
    func executeRequest() -> AnyPublisher<OMG, Error> {
        repository.request()
    }
    
}

import Foundation
import Combine

// MARK: - ViewModel
enum AsyncAwaitViewModelError: Error {
    case emptyInstance
}

protocol AsyncAwaitViewModel: OMGMemberDataSource {
    var requestErrorPublisher: AnyPublisher<Error, Never> { get }
    var omgMemeberListViewModelPublisher: AnyPublisher<[OMGMemberListViewModel], Never> { get }
    
    func didPressedRequestButton() async
    func didPressedRequestButton()
    func didPressedRequestButtonWithCombine() -> AnyCancellable
}

final class DefaultAsyncAwaitViewModel: AsyncAwaitViewModel {
    
    private let useCase: AsyncAwaitUseCase
    
    private var omg: OMG
    private var omgMember: [OMGMember]
    private var omgMemberListViewModel: [OMGMemberListViewModel]
    private let omgMemberListViewModelSubject: CurrentValueSubject<[OMGMemberListViewModel], Never>
    private let requestError: PassthroughSubject<Error, Never>

    var requestErrorPublisher: AnyPublisher<Error, Never> {
        requestError.eraseToAnyPublisher()
    }
    
    var omgMemeberListViewModelPublisher: AnyPublisher<[OMGMemberListViewModel], Never> {
        omgMemberListViewModelSubject.eraseToAnyPublisher()
    }
    
    init() {
        self.useCase = DefaultAsyncAwaitUseCase()
        self.omg = .init(member: [])
        self.omgMember = []
        self.omgMemberListViewModel = .init([])
        self.omgMemberListViewModelSubject = .init([])
        self.requestError = .init()
    }
    
    func didPressedRequestButton() async {
        do {
            let omg = try await useCase.executeRequest()
            self.omg = omg
            self.omgMember = omg.member
            self.omgMemberListViewModel = self.omgMember.map { .init(omgMember: $0) }
            self.omgMemberListViewModelSubject.send(omgMemberListViewModel)
        } catch let error {
            requestError.send(error)
        }
    }
    
    func didPressedRequestButton() {
        useCase.executeRequest { result in
            switch result {
            case .success(let omg):
                self.omg = omg
                self.omgMember = omg.member
                self.omgMemberListViewModel = self.omgMember.map { .init(omgMember: $0) }
                self.omgMemberListViewModelSubject.send(self.omgMemberListViewModel)
                
            case .failure(let error):
                self.requestError.send(error)
                
            }
        }
    }
    
    func didPressedRequestButtonWithCombine() -> AnyCancellable {
        let cancellable = useCase.executeRequest()
            .sink { [weak self] completion in
                switch completion {
                case .finished:
                    return
                    
                case .failure(let error):
                    self?.requestError.send(error)
                    
                }
            } receiveValue: { [weak self] omg in
                guard let self = self else {
                    self?.requestError.send(AsyncAwaitViewModelError.emptyInstance)
                    return
                }
                self.omg = omg
                self.omgMember = omg.member
                self.omgMemberListViewModel = self.omgMember.map { .init(omgMember: $0) }
                self.omgMemberListViewModelSubject.send(omgMemberListViewModel)
            }
        return cancellable
    }
    
    func didPressedRequestButtonWithAssign() {
        let cancellable = useCase.executeRequest()
    }
    
}

extension DefaultAsyncAwaitViewModel: OMGMemberDataSource {
    func numberOfMember() -> Int {
        return omgMemberListViewModel.count
    }
    
    func loadMember(at index: IndexPath) -> OMGMemberListViewModel {
        return omgMemberListViewModel[index.row]
    }
}

import Foundation

struct OMGMemberListViewModel {
    let name: String
}

extension OMGMemberListViewModel {
    init(omgMember: OMGMember) {
        self.name = omgMember.name
    }
}

import UIKit
import Combine

// MARK: - View
final class AsyncAwaitViewController: UIViewController {
    
    private let requestButton: UIButton = {
        let button = UIButton()
        button.translatesAutoresizingMaskIntoConstraints = false
        button.setTitle("Request", for: .normal)
        button.setTitleColor(.black, for: .normal)
        return button
    }()
    
    private let omgMemberListTableView: UITableView = {
        let tableView = UITableView()
        tableView.translatesAutoresizingMaskIntoConstraints = false
        return tableView
    }()
    
    private let viewModel: AsyncAwaitViewModel = DefaultAsyncAwaitViewModel()
    private var omgMemberListAdapter: OMGMemberListAdapter?
    
    private var cancellables: Set<AnyCancellable> = []
    
    override func viewDidLoad() {
        super.viewDidLoad()
        view.backgroundColor = .white
        
        addSubViews()
        
        setRequestButtonLayout()
        setOMGMemberListTableViewLayout()
        
        omgMemberListAdapter = OMGMemberListAdapter(tableView: omgMemberListTableView, dataSource: viewModel, delegate: self)
        
        addActionForRequestButton()
        
        subscribeOMG(from: viewModel.omgMemeberListViewModelPublisher)
        subscribeRequestError(from: viewModel.requestErrorPublisher)
    }
    
    private func addActionForRequestButton() {
        requestButton.addTarget(self, action: #selector(requestButtonAction), for: .touchUpInside)
    }
    
    @objc func requestButtonAction(_ sender: UIButton) {
        // MARK: - Async, Await
//        Task {
//            await viewModel.didPressedRequestButton()
//        }
        // MARK: - Completion
//        viewModel.didPressedRequestButton()
        // MARK: - Combine
        viewModel.didPressedRequestButtonWithCombine()
            .store(in: &cancellables)
    }
    
    private func presentAlert(of error: Error) {
        DispatchQueue.main.async {
            let alert = UIAlertController(title: "Error", message: "\(error)", preferredStyle: UIAlertController.Style.alert)
            let addAlertAction = UIAlertAction(title: "OK", style: .default)
            alert.addAction(addAlertAction)
            self.present(alert, animated: true, completion: nil)
        }
    }
    
    private func subscribeOMG(from omgPublisher: AnyPublisher<[OMGMemberListViewModel], Never>) {
        omgPublisher
            .receive(on: DispatchQueue.main)
            .sink { [weak self] omg in
                self?.omgMemberListTableView.reloadData()
            }
            .store(in: &cancellables)
    }
    
    private func subscribeRequestError(from requestErrorPublisher: AnyPublisher<Error, Never>) {
        requestErrorPublisher
            .receive(on: DispatchQueue.main)
            .sink { [weak self] requestError in
                self?.presentAlert(of: requestError)
            }
            .store(in: &cancellables)
    }
    
    private func addSubViews() {
        view.addSubview(requestButton)
        view.addSubview(omgMemberListTableView)
    }
    
    private func setOMGMemberListTableViewLayout() {
        omgMemberListTableView.leadingAnchor.constraint(equalTo: view.leadingAnchor).isActive = true
        omgMemberListTableView.topAnchor.constraint(equalTo: requestButton.bottomAnchor).isActive = true
        omgMemberListTableView.trailingAnchor.constraint(equalTo: view.trailingAnchor).isActive = true
        omgMemberListTableView.bottomAnchor.constraint(equalTo: view.bottomAnchor).isActive = true
    }
    
    private func setRequestButtonLayout() {
        requestButton.topAnchor.constraint(equalTo: view.topAnchor, constant: 120).isActive = true
        requestButton.centerXAnchor.constraint(equalTo: view.centerXAnchor).isActive = true
    }
    
}

import UIKit

final class OMGListTableViewCell: UITableViewCell {
    
    private let nameLabel: UILabel = {
        let label = UILabel()
        label.translatesAutoresizingMaskIntoConstraints = false
        label.textColor = .black
        return label
    }()
    
    override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)
        addSubview(nameLabel)
        nameLabel.leadingAnchor.constraint(equalTo: leadingAnchor).isActive = true
        nameLabel.centerYAnchor.constraint(equalTo: centerYAnchor).isActive = true
    }
    
    required init?(coder: NSCoder) {
        fatalError()
    }
    
    func apply(viewModel: OMGMemberListViewModel) {
        nameLabel.text = viewModel.name
    }
    
}

extension AsyncAwaitViewController: OMGMemberDelegate {
    func heightForRow(at indexPath: IndexPath) -> CGFloat {
        return 120
    }
}

import UIKit

protocol OMGMemberDataSource: AnyObject {
    func numberOfMember() -> Int
    func loadMember(at index: IndexPath) -> OMGMemberListViewModel
    
}

protocol OMGMemberDelegate: AnyObject {
    func heightForRow(at indexPath: IndexPath) -> CGFloat
}

final class OMGMemberListAdapter: NSObject {
    
    private let tableView: UITableView
    private weak var dataSource: OMGMemberDataSource?
    private weak var delegate: OMGMemberDelegate?

    init(tableView: UITableView, dataSource: OMGMemberDataSource?, delegate: OMGMemberDelegate) {
        tableView.register(OMGListTableViewCell.self, forCellReuseIdentifier: "OMGListTableViewCell")
        
        self.tableView = tableView
        self.dataSource = dataSource
        self.delegate = delegate
        super.init()
        
        tableView.dataSource = self
        tableView.delegate = self
    }
    
}

extension OMGMemberListAdapter: UITableViewDataSource {
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        guard let dataSource = dataSource else { return .init() }
        return dataSource.numberOfMember()
    }
    
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        guard let cell = tableView.dequeueReusableCell(withIdentifier: "OMGListTableViewCell", for: indexPath) as? OMGListTableViewCell else { return .init() }
        guard let dataSource = dataSource else { return .init() }
        let member = dataSource.loadMember(at: indexPath)
        cell.apply(viewModel: member)
        return cell
    }
}

extension OMGMemberListAdapter: UITableViewDelegate {
    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        return delegate?.heightForRow(at: indexPath) ?? 0
    }
}

Simulator

0개의 댓글