diff --git a/iOS/Layover/Layover.xcodeproj/project.pbxproj b/iOS/Layover/Layover.xcodeproj/project.pbxproj index 2c80ebc..9288b24 100644 --- a/iOS/Layover/Layover.xcodeproj/project.pbxproj +++ b/iOS/Layover/Layover.xcodeproj/project.pbxproj @@ -113,6 +113,10 @@ 19AE482A2B2A127E00DD4612 /* HLSAssetResourceLoaderDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19AE48292B2A127E00DD4612 /* HLSAssetResourceLoaderDelegate.swift */; }; 19AE482C2B2A1A8B00DD4612 /* HLSSliceResourceLoader.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19AE482B2B2A1A8B00DD4612 /* HLSSliceResourceLoader.swift */; }; 19AE482E2B2A24C700DD4612 /* URL+.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19AE482D2B2A24C700DD4612 /* URL+.swift */; }; + 19B665D92B4EEDDD0083E63C /* SignUpWorkerTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19B665D52B4EEDDD0083E63C /* SignUpWorkerTests.swift */; }; + 19B665DA2B4EEDDD0083E63C /* SignUpInteractorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19B665D62B4EEDDD0083E63C /* SignUpInteractorTests.swift */; }; + 19B665DB2B4EEDDD0083E63C /* SignUpPresenterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19B665D72B4EEDDD0083E63C /* SignUpPresenterTests.swift */; }; + 19B665DD2B4F0A7B0083E63C /* MockSignUpWorker.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19B665DC2B4F0A7B0083E63C /* MockSignUpWorker.swift */; }; 19C7AFCE2B02410F003B35F2 /* AuthManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19C7AFCD2B02410F003B35F2 /* AuthManager.swift */; }; 19C7AFD62B02584D003B35F2 /* KeychainStored.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19C7AFD52B02584D003B35F2 /* KeychainStored.swift */; }; 19E79AC02B0A85D0009EA9ED /* LoopingPlayerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 19E79ABF2B0A85D0009EA9ED /* LoopingPlayerView.swift */; }; @@ -371,6 +375,10 @@ 19AE48292B2A127E00DD4612 /* HLSAssetResourceLoaderDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = HLSAssetResourceLoaderDelegate.swift; sourceTree = ""; }; 19AE482B2B2A1A8B00DD4612 /* HLSSliceResourceLoader.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = HLSSliceResourceLoader.swift; sourceTree = ""; }; 19AE482D2B2A24C700DD4612 /* URL+.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "URL+.swift"; sourceTree = ""; }; + 19B665D52B4EEDDD0083E63C /* SignUpWorkerTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SignUpWorkerTests.swift; sourceTree = ""; }; + 19B665D62B4EEDDD0083E63C /* SignUpInteractorTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SignUpInteractorTests.swift; sourceTree = ""; }; + 19B665D72B4EEDDD0083E63C /* SignUpPresenterTests.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SignUpPresenterTests.swift; sourceTree = ""; }; + 19B665DC2B4F0A7B0083E63C /* MockSignUpWorker.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = MockSignUpWorker.swift; sourceTree = ""; }; 19C7AFCD2B02410F003B35F2 /* AuthManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AuthManager.swift; sourceTree = ""; }; 19C7AFD52B02584D003B35F2 /* KeychainStored.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = KeychainStored.swift; sourceTree = ""; }; 19E79ABF2B0A85D0009EA9ED /* LoopingPlayerView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LoopingPlayerView.swift; sourceTree = ""; }; @@ -626,6 +634,7 @@ isa = PBXGroup; children = ( 8363A32B2B4C1C6900772DDF /* Playback */, + 19B665D32B4EEDDD0083E63C /* SignUp */, 19AE481D2B29D02700DD4612 /* EditProfile */, 19AE48122B28C2A800DD4612 /* Setting */, 192513632B26F7BB001533FA /* TagPlayList */, @@ -655,6 +664,7 @@ 192513842B27852C001533FA /* MockUserWorker.swift */, 19AE481B2B28C53800DD4612 /* MockSettingWorker.swift */, 8363A3322B4D6E9B00772DDF /* MockPlaybackWorker.swift */, + 19B665DC2B4F0A7B0083E63C /* MockSignUpWorker.swift */, ); path = Workers; sourceTree = ""; @@ -801,6 +811,16 @@ path = HLSResourceLoader; sourceTree = ""; }; + 19B665D32B4EEDDD0083E63C /* SignUp */ = { + isa = PBXGroup; + children = ( + 19B665D62B4EEDDD0083E63C /* SignUpInteractorTests.swift */, + 19B665D72B4EEDDD0083E63C /* SignUpPresenterTests.swift */, + 19B665D52B4EEDDD0083E63C /* SignUpWorkerTests.swift */, + ); + path = SignUp; + sourceTree = ""; + }; 19BB8A572B07BEE30070B922 /* UIComponents */ = { isa = PBXGroup; children = ( @@ -1597,6 +1617,7 @@ files = ( 192513852B27852C001533FA /* MockUserWorker.swift in Sources */, 192513832B277CD7001533FA /* ProfilePresenterTests.swift in Sources */, + 19B665DD2B4F0A7B0083E63C /* MockSignUpWorker.swift in Sources */, 192513812B277CD7001533FA /* ProfileInteractorTests.swift in Sources */, 194C21C42B1DEE6B00C62645 /* HomeInteractorTests.swift in Sources */, FC4E0C0F2B282AE500152596 /* UploadPostPresenterTests.swift in Sources */, @@ -1618,12 +1639,15 @@ 1925136D2B26F84E001533FA /* MockTagPlayListWorker.swift in Sources */, 19AE481A2B28C2B700DD4612 /* SettingPresenterTests.swift in Sources */, 19AE48172B28C2B700DD4612 /* SettingViewControllerTests.swift in Sources */, + 19B665DA2B4EEDDD0083E63C /* SignUpInteractorTests.swift in Sources */, + 19B665DB2B4EEDDD0083E63C /* SignUpPresenterTests.swift in Sources */, 194C21C32B1DEE6B00C62645 /* HomeViewControllerTests.swift in Sources */, 8363A32F2B4C329100772DDF /* PlaybackInteractorTests.swift in Sources */, 192513692B26F7CE001533FA /* TagPlayListInteractorTests.swift in Sources */, 19AE48232B29D03D00DD4612 /* EditProfileInteractorTests.swift in Sources */, 194C21CC2B1DF39200C62645 /* MockHomeWorker.swift in Sources */, FC4E0C0E2B282AE500152596 /* UploadPostWorkerTests.swift in Sources */, + 19B665D92B4EEDDD0083E63C /* SignUpWorkerTests.swift in Sources */, FC4E0C112B28595200152596 /* MockUploadPostWorker.swift in Sources */, FC4E0C0D2B282AE500152596 /* UploadPostInteractorTests.swift in Sources */, ); diff --git a/iOS/Layover/Layover/Scenes/Map/MapViewController.swift b/iOS/Layover/Layover/Scenes/Map/MapViewController.swift index d656e8e..6e7f7ea 100644 --- a/iOS/Layover/Layover/Scenes/Map/MapViewController.swift +++ b/iOS/Layover/Layover/Scenes/Map/MapViewController.swift @@ -161,7 +161,7 @@ final class MapViewController: BaseViewController { let maximumZoomScale: CGFloat = 1.0 let inset = (screenSize.width - screenSize.width * groupWidthDimension) / 2 let section: NSCollectionLayoutSection = .makeCarouselSection(groupWidthDimension: groupWidthDimension) - section.orthogonalScrollingBehavior = .groupPagingCentered + section.orthogonalScrollingBehavior = .groupPaging section.interGroupSpacing = 0 section.contentInsets = NSDirectionalEdgeInsets(top: 0, leading: inset, @@ -173,12 +173,11 @@ final class MapViewController: BaseViewController { let distanceFromCenter = abs((item.center.x - offset.x) - environment.container.contentSize.width / 2.0) let scale = max(maximumZoomScale - (distanceFromCenter / containerWidth), minumumZoomScale) item.transform = CGAffineTransform(scaleX: scale, y: scale) - let cell = self?.carouselCollectionView.cellForItem(at: item.indexPath) as? MapCarouselCollectionViewCell + guard let cell = self?.carouselCollectionView.cellForItem(at: item.indexPath) as? MapCarouselCollectionViewCell else { return } if scale >= maximumZoomScale * 0.9 { - cell?.play() self?.selectAnnotation(at: item.indexPath) } else { - cell?.pause() + cell.pause() } } } @@ -197,7 +196,6 @@ final class MapViewController: BaseViewController { carouselCollectionViewHeight.constant = isSelected ? 151 : 0 UIView.animate(withDuration: 0.3) { annotationView.transform = isSelected ? CGAffineTransform(scaleX: 1.3, y: 1.3) : .identity - self.view.layoutIfNeeded() } } @@ -256,15 +254,18 @@ extension MapViewController: MKMapViewDelegate { if let annotaion = annotation as? LOAnnotation { // 선택된 pin 정보와 datasource를 비교해 selected item을 찾음 let snapshot = carouselDatasource.snapshot() - guard let selectedItemIdentifiers = carouselDatasource.snapshot().itemIdentifiers.filter({ post in + guard let selectedItemIdentifiers = snapshot.itemIdentifiers.filter({ post in return post.boardID == annotaion.boardID }).first else { return } guard let section = snapshot.sectionIdentifier(containingItem: selectedItemIdentifiers), let itemIndex = snapshot.indexOfItem(selectedItemIdentifiers), let sectionIndex = snapshot.indexOfSection(section) else { return } - carouselCollectionView.scrollToItem(at: IndexPath(item: itemIndex, section: sectionIndex), + let indexPath = IndexPath(item: itemIndex, section: sectionIndex) + carouselCollectionView.scrollToItem(at: indexPath, at: .centeredHorizontally, animated: false) + guard let cell = carouselCollectionView.cellForItem(at: indexPath) as? MapCarouselCollectionViewCell else { return } + cell.play() } } diff --git a/iOS/Layover/Layover/Scenes/Map/Views/MapCarouselCollectionViewCell.swift b/iOS/Layover/Layover/Scenes/Map/Views/MapCarouselCollectionViewCell.swift index 5fb4243..408fb55 100644 --- a/iOS/Layover/Layover/Scenes/Map/Views/MapCarouselCollectionViewCell.swift +++ b/iOS/Layover/Layover/Scenes/Map/Views/MapCarouselCollectionViewCell.swift @@ -48,7 +48,9 @@ final class MapCarouselCollectionViewCell: UICollectionViewCell { spinner.stopAnimating() loopingPlayerView.disable() loopingPlayerView.prepareVideo(with: url, - timeRange: CMTimeRange(start: .zero, duration: CMTime(value: 1800, timescale: 600))) + assetResourceLoaderDelegate: HLSAssetResourceLoaderDelegate(resourceLoader: HLSSliceResourceLoader()), + loopStart: .zero, + duration: 3.0) loopingPlayerView.player?.isMuted = true } diff --git a/iOS/Layover/Layover/Scenes/SignUpScene/SignUpInteractor.swift b/iOS/Layover/Layover/Scenes/SignUpScene/SignUpInteractor.swift index dd98c7d..3177943 100644 --- a/iOS/Layover/Layover/Scenes/SignUpScene/SignUpInteractor.swift +++ b/iOS/Layover/Layover/Scenes/SignUpScene/SignUpInteractor.swift @@ -10,8 +10,8 @@ import UIKit protocol SignUpBusinessLogic { func validateNickname(with request: SignUpModels.ValidateNickname.Request) - func checkDuplication(with request: SignUpModels.CheckDuplication.Request) - func signUp(with request: SignUpModels.SignUp.Request) + func checkDuplication(with request: SignUpModels.CheckDuplication.Request) async + func signUp(with request: SignUpModels.SignUp.Request) async } protocol SignUpDataStore: AnyObject { @@ -40,33 +40,29 @@ final class SignUpInteractor: SignUpBusinessLogic, SignUpDataStore { presenter?.presentNicknameValidation(with: SignUpModels.ValidateNickname.Response(nicknameState: response)) } - func checkDuplication(with request: SignUpModels.CheckDuplication.Request) { - Task { - let response = await userWorker?.checkNotDuplication(for: request.nickname) - await MainActor.run { - presenter?.presentNicknameDuplication(with: SignUpModels.CheckDuplication.Response(isValid: response ?? false)) - } + func checkDuplication(with request: SignUpModels.CheckDuplication.Request) async { + let response = await userWorker?.checkNotDuplication(for: request.nickname) + await MainActor.run { + presenter?.presentNicknameDuplication(with: SignUpModels.CheckDuplication.Response(isValid: response ?? false)) } } // MARK: - UseCase: SignUp - func signUp(with request: SignUpModels.SignUp.Request) { + func signUp(with request: SignUpModels.SignUp.Request) async { guard let signUpType, let socialToken else { return } - Task { - switch signUpType { - case .kakao: - if await signUpWorker?.signUp(withKakao: socialToken, username: request.nickname) == true { - await MainActor.run { - presenter?.presentSignUpSuccess() - } + switch signUpType { + case .kakao: + if await signUpWorker?.signUp(withKakao: socialToken, username: request.nickname) == true { + await MainActor.run { + presenter?.presentSignUpSuccess() } - case .apple: - if await signUpWorker?.signUp(withApple: socialToken, username: request.nickname) == true { - await MainActor.run { - presenter?.presentSignUpSuccess() - } + } + case .apple: + if await signUpWorker?.signUp(withApple: socialToken, username: request.nickname) == true { + await MainActor.run { + presenter?.presentSignUpSuccess() } } } diff --git a/iOS/Layover/Layover/Scenes/SignUpScene/SignUpPresenter.swift b/iOS/Layover/Layover/Scenes/SignUpScene/SignUpPresenter.swift index 64a0c61..0cc3278 100644 --- a/iOS/Layover/Layover/Scenes/SignUpScene/SignUpPresenter.swift +++ b/iOS/Layover/Layover/Scenes/SignUpScene/SignUpPresenter.swift @@ -31,7 +31,7 @@ final class SignUpPresenter: SignUpPresentationLogic { func presentNicknameDuplication(with response: SignUpModels.CheckDuplication.Response) { let viewModel = Models.CheckDuplication.ViewModel(canSignUp: response.isValid) - viewController?.displayNickanmeDuplication(response: viewModel) + viewController?.displayNicknameDuplication(response: viewModel) } func presentSignUpSuccess() { diff --git a/iOS/Layover/Layover/Scenes/SignUpViewController.swift b/iOS/Layover/Layover/Scenes/SignUpViewController.swift index fe01214..bbaffa9 100644 --- a/iOS/Layover/Layover/Scenes/SignUpViewController.swift +++ b/iOS/Layover/Layover/Scenes/SignUpViewController.swift @@ -10,7 +10,7 @@ import UIKit protocol SignUpDisplayLogic: AnyObject { func displayNicknameValidation(response: SignUpModels.ValidateNickname.ViewModel) - func displayNickanmeDuplication(response: SignUpModels.CheckDuplication.ViewModel) + func displayNicknameDuplication(response: SignUpModels.CheckDuplication.ViewModel) func navigateToMain() } @@ -137,7 +137,9 @@ final class SignUpViewController: BaseViewController { @objc private func checkDuplicateNicknameButtonDidTap(_ sender: UIButton) { guard let nickname = nicknameTextfield.text else { return } checkDuplicateNicknameButton.isEnabled = false - interactor?.checkDuplication(with: SignUpModels.CheckDuplication.Request(nickname: nickname)) + Task { + await interactor?.checkDuplication(with: SignUpModels.CheckDuplication.Request(nickname: nickname)) + } } @objc private func popViewController() { @@ -146,7 +148,9 @@ final class SignUpViewController: BaseViewController { @objc private func signUpButtonDidTap(_ sender: UIButton) { guard let nickname = nicknameTextfield.text else { return } - interactor?.signUp(with: SignUpModels.SignUp.Request(nickname: nickname)) + Task { + await interactor?.signUp(with: SignUpModels.SignUp.Request(nickname: nickname)) + } } } @@ -159,7 +163,7 @@ extension SignUpViewController: SignUpDisplayLogic { nicknameAlertLabel.textColor = .error } - func displayNickanmeDuplication(response: SignUpModels.CheckDuplication.ViewModel) { + func displayNicknameDuplication(response: SignUpModels.CheckDuplication.ViewModel) { nicknameAlertLabel.isHidden = false nicknameAlertLabel.text = response.alertDescription nicknameAlertLabel.textColor = response.canSignUp ? .correct : .error diff --git a/iOS/Layover/Layover/Workers/Mocks/MockSignUpWorker.swift b/iOS/Layover/Layover/Workers/Mocks/MockSignUpWorker.swift index 2bfc300..6198cf3 100644 --- a/iOS/Layover/Layover/Workers/Mocks/MockSignUpWorker.swift +++ b/iOS/Layover/Layover/Workers/Mocks/MockSignUpWorker.swift @@ -9,7 +9,7 @@ import Foundation import OSLog -final class MockSignUpWorker { +class MockSignUpWorker: SignUpWorkerProtocol { // MARK: - Properties @@ -21,11 +21,6 @@ final class MockSignUpWorker { authManager: StubAuthManager())) { self.provider = provider } -} - -// MARK: - SignUpWorkerProtocol - -extension MockSignUpWorker: SignUpWorkerProtocol { func signUp(withKakao socialToken: String, username: String) async -> Bool { guard let mockFileLocation = Bundle.main.url(forResource: "LoginData", withExtension: "json"), @@ -86,5 +81,4 @@ extension MockSignUpWorker: SignUpWorkerProtocol { return false } } - } diff --git a/iOS/Layover/LayoverTests/Mocks/Workers/MockSignUpWorker.swift b/iOS/Layover/LayoverTests/Mocks/Workers/MockSignUpWorker.swift new file mode 100644 index 0000000..07eb3e8 --- /dev/null +++ b/iOS/Layover/LayoverTests/Mocks/Workers/MockSignUpWorker.swift @@ -0,0 +1,85 @@ +// +// MockSignUpWorker.swift +// LayoverTests +// +// Created by 김인환 on 1/11/24. +// Copyright © 2024 CodeBomber. All rights reserved. +// + +@testable import Layover +import Foundation +import OSLog + +class MockSignUpWorker: SignUpWorkerProtocol { + + // MARK: - Properties + + private let provider: ProviderType + + // MARK: - Initializer + + init(provider: ProviderType = Provider(session: .initMockSession(), + authManager: StubAuthManager())) { + self.provider = provider + } + + func signUp(withKakao socialToken: String, username: String) async -> Bool { + guard let mockFileLocation = Bundle(for: type(of: self)).url(forResource: "LoginData", withExtension: "json"), + let mockData = try? Data(contentsOf: mockFileLocation) else { + return false + } + + MockURLProtocol.requestHandler = { request in + let response = HTTPURLResponse(url: request.url!, + statusCode: 200, + httpVersion: nil, + headerFields: nil) + return (response, mockData, nil) + } + + do { + var bodyParameters = [String: String]() + bodyParameters.updateValue(socialToken, forKey: "accessToken") + bodyParameters.updateValue(username, forKey: "username") + + let endPoint = EndPoint>(path: "/oauth/signup/kakao", + method: .POST, + bodyParameters: bodyParameters) + let response = try await provider.request(with: endPoint, authenticationIfNeeded: false, retryCount: 0) + return true + } catch { + os_log(.error, log: .data, "%@", error.localizedDescription) + return false + } + } + + func signUp(withApple identityToken: String, username: String) async -> Bool { + guard let fileLocation: URL = Bundle(for: type(of: self)).url(forResource: "LoginData", withExtension: "json") else { + return false + } + guard let mockData: Data = try? Data(contentsOf: fileLocation) else { + return false + } + MockURLProtocol.requestHandler = { request in + let response = HTTPURLResponse(url: request.url!, + statusCode: 200, + httpVersion: nil, + headerFields: nil) + return (response, mockData, nil) + } + do { + var bodyParameters: [String: String] = [:] + bodyParameters.updateValue(identityToken, forKey: "identityToken") + bodyParameters.updateValue(username, forKey: "username") + + let endPoint = EndPoint>(path: "/oauth/signup/apple", + method: .POST, + bodyParameters: bodyParameters) + let response = try await provider.request(with: endPoint, authenticationIfNeeded: false, retryCount: 0) + return true + } catch { + os_log(.error, log: .data, "%@", error.localizedDescription) + return false + } + } +} diff --git a/iOS/Layover/LayoverTests/Mocks/Workers/MockUserWorker.swift b/iOS/Layover/LayoverTests/Mocks/Workers/MockUserWorker.swift index 117dfb1..5faa754 100644 --- a/iOS/Layover/LayoverTests/Mocks/Workers/MockUserWorker.swift +++ b/iOS/Layover/LayoverTests/Mocks/Workers/MockUserWorker.swift @@ -9,7 +9,7 @@ import Foundation import OSLog -final class MockUserWorker: UserWorkerProtocol { +class MockUserWorker: UserWorkerProtocol { // MARK: - Properties diff --git a/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpInteractorTests.swift b/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpInteractorTests.swift new file mode 100644 index 0000000..7d83d66 --- /dev/null +++ b/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpInteractorTests.swift @@ -0,0 +1,159 @@ +// +// SignUpInteractorTests.swift +// Layover +// +// Created by 김인환 on 1/6/24. +// Copyright (c) 2024 CodeBomber. All rights reserved. +// +// This file was generated by the Clean Swift Xcode Templates so +// you can apply clean architecture to your iOS and Mac projects, +// see http://clean-swift.com +// + +@testable import Layover +import XCTest + +final class SignUpInteractorTests: XCTestCase { + // MARK: Subject under test + + var sut: SignUpInteractor! + + typealias Models = SignUpModels + + // MARK: - Test lifecycle + + override func setUp() { + super.setUp() + setupSignUpInteractor() + } + + // MARK: - Test setup + + func setupSignUpInteractor() { + sut = SignUpInteractor() + } + + // MARK: - Test doubles + + final class SignUpPresentationLogicSpy: SignUpPresentationLogic { + var presentSignUpSuccessDidCalled = false + var presentNicknameValidationDidCalled = false + var presentNicknameValidationResponse: Models.ValidateNickname.Response! + var presentNicknameDuplicationDidCalled = false + var presentNicknameDupllicationResponse: Models.CheckDuplication.Response! + + func presentSignUpSuccess() { + presentSignUpSuccessDidCalled = true + } + + func presentNicknameValidation(with response: Models.ValidateNickname.Response) { + presentNicknameValidationDidCalled = true + presentNicknameValidationResponse = response + } + + func presentNicknameDuplication(with response: Models.CheckDuplication.Response) { + presentNicknameDuplicationDidCalled = true + presentNicknameDupllicationResponse = response + } + } + + final class UserWorkerSpy: MockUserWorker { + var validateNicknameCalled = false + var checkNotDuplicationCalled = false + + override func validateNickname(_ nickname: String) -> NicknameState { + validateNicknameCalled = true + return super.validateNickname(nickname) + } + + override func checkNotDuplication(for userName: String) async -> Bool? { + checkNotDuplicationCalled = true + return await super.checkNotDuplication(for: userName) + } + } + + final class SignUpWorkerSpy: MockSignUpWorker { + var signUpWithKakaoCalled = false + var signUpWithAppleCalled = false + + override func signUp(withKakao token: String, username: String) async -> Bool { + signUpWithKakaoCalled = true + return await super.signUp(withKakao: token, username: username) + } + + override func signUp(withApple identityToken: String, username: String) async -> Bool { + signUpWithAppleCalled = true + return await super.signUp(withApple: identityToken, username: username) + } + } + + // MARK: - Tests + + func test_유효한_닉네임값으로_validateNickname을_호출하면_UserWorker를_통해_닉네임_유효성_검사를_요청하고_valid_결과를_presenter의_presentNicknameValidation를_호출한다() { + // arrange + let presenterSpy = SignUpPresentationLogicSpy() + sut.presenter = presenterSpy + let userWorkerSpy = UserWorkerSpy() + sut.userWorker = userWorkerSpy + + // act + sut.validateNickname(with: Models.ValidateNickname.Request(nickname: "안유진")) + + // assert + XCTAssertTrue(userWorkerSpy.validateNicknameCalled) + XCTAssertTrue(presenterSpy.presentNicknameValidationDidCalled) + XCTAssertNotNil(presenterSpy.presentNicknameValidationResponse.nicknameState) + } + + func test_checkDuplication을_호출하면_UserWorker를_통해_닉네임_중복_검사를_요청하고_결과를_presenter의_presentNicknameDuplication를_호출하여_전달한다() async { + // arrange + let presenterSpy = SignUpPresentationLogicSpy() + sut.presenter = presenterSpy + let userWorkerSpy = UserWorkerSpy() + sut.userWorker = userWorkerSpy + let signUpWorkerSpy = SignUpWorkerSpy() + sut.signUpWorker = signUpWorkerSpy + + // act + await sut.checkDuplication(with: Models.CheckDuplication.Request(nickname: "안유진")) + + // assert + XCTAssertTrue(userWorkerSpy.checkNotDuplicationCalled) + XCTAssertTrue(presenterSpy.presentNicknameDuplicationDidCalled) + XCTAssertNotNil(presenterSpy.presentNicknameDupllicationResponse.isValid) + } + + func test_카카오_signUp을_호출하면_SignUpWorker를_통해_회원가입을_요청하고_presenter의_presentSignUpSuccess를_호출한다() async { + // arrange + let presenterSpy = SignUpPresentationLogicSpy() + sut.presenter = presenterSpy + let signUpWorkerSpy = SignUpWorkerSpy() + signUpWorkerSpy.signUpWithKakaoCalled = true + sut.signUpWorker = signUpWorkerSpy + sut.signUpType = .kakao + sut.socialToken = "1234" + + // act + await sut.signUp(with: Models.SignUp.Request(nickname: "안유진")) + + // assert + XCTAssertTrue(presenterSpy.presentSignUpSuccessDidCalled) + } + + func test_애플_signUp을_호출하면_SignUpWorker를_통해_회원가입을_요청하고_presenter의_presentSignUpSuccess를_호출한다() async { + // arrange + let presenterSpy = SignUpPresentationLogicSpy() + sut.presenter = presenterSpy + let signUpWorkerSpy = SignUpWorkerSpy() + signUpWorkerSpy.signUpWithAppleCalled = true + sut.signUpWorker = signUpWorkerSpy + sut.signUpType = .apple + sut.socialToken = "1234" + + // act + await sut.signUp(with: Models.SignUp.Request(nickname: "안유진")) + + // assert + XCTAssertTrue(presenterSpy.presentSignUpSuccessDidCalled) + } +} diff --git a/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpPresenterTests.swift b/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpPresenterTests.swift new file mode 100644 index 0000000..dd92fb6 --- /dev/null +++ b/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpPresenterTests.swift @@ -0,0 +1,144 @@ +// +// SignUpPresenterTests.swift +// Layover +// +// Created by 김인환 on 1/6/24. +// Copyright (c) 2024 CodeBomber. All rights reserved. +// +// This file was generated by the Clean Swift Xcode Templates so +// you can apply clean architecture to your iOS and Mac projects, +// see http://clean-swift.com +// + +@testable import Layover +import XCTest + +final class SignUpPresenterTests: XCTestCase { + // MARK: Subject under test + + var sut: SignUpPresenter! + + typealias Models = SignUpModels + + // MARK: - Test lifecycle + + override func setUp() { + super.setUp() + setupSignUpPresenter() + } + + // MARK: - Test setup + + func setupSignUpPresenter() { + sut = SignUpPresenter() + } + + // MARK: - Test doubles + + final class SignUpDisplayLogicSpy: SignUpDisplayLogic { + var displayNicknameValidationDidCalled = false + var displayNicknameValidationResponse: Models.ValidateNickname.ViewModel! + var displayNicknameDuplicationDidCalled = false + var displayNicknameDuplicationResponse: Models.CheckDuplication.ViewModel! + var navigateToMainDidCalled = false + + func displayNicknameValidation(response: Models.ValidateNickname.ViewModel) { + displayNicknameValidationDidCalled = true + displayNicknameValidationResponse = response + } + + func displayNicknameDuplication(response: Models.CheckDuplication.ViewModel) { + displayNicknameDuplicationDidCalled = true + displayNicknameDuplicationResponse = response + } + + func navigateToMain() { + navigateToMainDidCalled = true + } + } + + // MARK: - Tests + + func test_presentNicknameValidation가_호출되고_전달되는_nicknameState가_valid면_ViewController의_displayNicknameValidation이_호출되고_올바른_값이_전달된다() { + // arrange + let spy = SignUpDisplayLogicSpy() + sut.viewController = spy + let nicknameState = NicknameState.valid + + // act + sut.presentNicknameValidation(with: Models.ValidateNickname.Response(nicknameState: nicknameState)) + + // assert + XCTAssertTrue(spy.displayNicknameValidationDidCalled) + XCTAssertTrue(spy.displayNicknameValidationResponse.canCheckDuplication) + XCTAssertEqual(spy.displayNicknameValidationResponse.alertDescription, nicknameState.description) + } + + func test_presentNicknameValidation가_호출되고_전달되는_nicknameState가_invalidCharacter면_ViewController의_displayNicknameValidation이_호출되고_올바른_값이_전달된다() { + // arrange + let spy = SignUpDisplayLogicSpy() + sut.viewController = spy + let nicknameState = NicknameState.invalidCharacter + + // act + sut.presentNicknameValidation(with: Models.ValidateNickname.Response(nicknameState: nicknameState)) + + // assert + XCTAssertTrue(spy.displayNicknameValidationDidCalled) + XCTAssertFalse(spy.displayNicknameValidationResponse.canCheckDuplication) + XCTAssertEqual(spy.displayNicknameValidationResponse.alertDescription, nicknameState.description) + } + + func test_presentNicknameValidation가_호출되고_전달되는_nicknameState가_lessThan2GreaterThan8면_ViewController의_displayNicknameValidation이_호출되고_올바른_값이_전달된다() { + // arrange + let spy = SignUpDisplayLogicSpy() + sut.viewController = spy + let nicknameState = NicknameState.lessThan2GreaterThan8 + + // act + sut.presentNicknameValidation(with: Models.ValidateNickname.Response(nicknameState: .lessThan2GreaterThan8)) + + // assert + XCTAssertTrue(spy.displayNicknameValidationDidCalled) + XCTAssertFalse(spy.displayNicknameValidationResponse.canCheckDuplication) + XCTAssertEqual(spy.displayNicknameValidationResponse.alertDescription, nicknameState.description) + } + + func test_presentNicknameDuplication가_호출되면_ViewController의_displayNicknameDuplication이_호출되고_isValid_true값을_그대로_전달한다() { + // arrange + let spy = SignUpDisplayLogicSpy() + sut.viewController = spy + + // act + sut.presentNicknameDuplication(with: Models.CheckDuplication.Response(isValid: true)) + + // assert + XCTAssertTrue(spy.displayNicknameDuplicationDidCalled) + XCTAssertTrue(spy.displayNicknameDuplicationResponse.canSignUp) + } + + func test_presentNicknameDuplication가_호출되면_ViewController의_displayNicknameDuplication이_호출되고_전달받은_isValid_false값을_그대로_전달한다() { + // arrange + let spy = SignUpDisplayLogicSpy() + sut.viewController = spy + + // act + sut.presentNicknameDuplication(with: Models.CheckDuplication.Response(isValid: false)) + + // assert + XCTAssertTrue(spy.displayNicknameDuplicationDidCalled) + XCTAssertFalse(spy.displayNicknameDuplicationResponse.canSignUp) + } + + func test_presentSignUpSuccess가_호출되면_ViewController의_navigateToMain이_호출된다() { + // arrange + let spy = SignUpDisplayLogicSpy() + sut.viewController = spy + + // act + sut.presentSignUpSuccess() + + // assert + XCTAssertTrue(spy.navigateToMainDidCalled) + } +} diff --git a/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpWorkerTests.swift b/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpWorkerTests.swift new file mode 100644 index 0000000..7092743 --- /dev/null +++ b/iOS/Layover/LayoverTests/Scenes/SignUp/SignUpWorkerTests.swift @@ -0,0 +1,179 @@ +// +// SignUpWorkerTests.swift +// Layover +// +// Created by 김인환 on 1/6/24. +// Copyright (c) 2024 CodeBomber. All rights reserved. +// +// This file was generated by the Clean Swift Xcode Templates so +// you can apply clean architecture to your iOS and Mac projects, +// see http://clean-swift.com +// + +@testable import Layover +import XCTest + +final class SignUpWorkerTests: XCTestCase { + // MARK: Subject under test + + var sut: SignUpWorker! + var authManagerSpy: AuthManagerSpy! + + // MARK: - Test lifecycle + + override func setUp() { + super.setUp() + setupSignUpWorker() + } + + override func tearDown() { + super.tearDown() + } + + // MARK: - Test setup + + func setupSignUpWorker() { + authManagerSpy = AuthManagerSpy() + sut = SignUpWorker(provider: Provider(session: .initMockSession(), authManager: StubAuthManager()), + authManager: authManagerSpy) + } + + // MARK: - Test Doubles + + final class AuthManagerSpy: StubAuthManager { + var loginCalled = false + var logoutCalled = false + + override init() { + super.init() + self.accessToken = nil + self.refreshToken = nil + self.loginType = nil + self.isLoggedIn = nil + } + + override func login(accessToken: String?, refreshToken: String?, loginType: LoginType?) { + loginCalled = true + super.login(accessToken: accessToken, refreshToken: refreshToken, loginType: loginType) + } + + override func logout() { + logoutCalled = true + super.logout() + } + } + + // MARK: - Tests + + func test_카카오_회원가입을_성공적으로_했을때_true를_반환하고_로그인_처리된다() async { + // arrange + guard let mockFileLocation = Bundle(for: type(of: self)).url(forResource: "LoginData", withExtension: "json"), + let mockData = try? Data(contentsOf: mockFileLocation) else { + XCTFail("Mock json 파일 로드 실패.") + return + } + + MockURLProtocol.requestHandler = { request in + let response = HTTPURLResponse(url: request.url!, + statusCode: 200, + httpVersion: nil, + headerFields: nil) + return (response, mockData, nil) + } + + // act + let isSuccess = await sut.signUp(withKakao: "MockToken", username: "안유진") + + // assert + XCTAssertTrue(isSuccess) + XCTAssertTrue(authManagerSpy.loginCalled) + XCTAssertEqual(authManagerSpy.accessToken, "mockAccessToken") + XCTAssertEqual(authManagerSpy.refreshToken, "mockRefreshToken") + XCTAssertEqual(authManagerSpy.loginType, .kakao) + XCTAssertTrue(authManagerSpy.isLoggedIn == true) + } + + func test_카카오_회원가입을_실패했을때_false를_반환하고_로그인_처리되지_않는다() async { + // arrange + guard let mockFileLocation = Bundle(for: type(of: self)).url(forResource: "LoginData", withExtension: "json"), + let mockData = try? Data(contentsOf: mockFileLocation) else { + XCTFail("Mock json 파일 로드 실패.") + return + } + + MockURLProtocol.requestHandler = { request in + let response = HTTPURLResponse(url: request.url!, + statusCode: 400, + httpVersion: nil, + headerFields: nil) + return (response, mockData, nil) + } + + // act + let isSuccess = await sut.signUp(withKakao: "MockToken", username: "안유진") + + // assert + XCTAssertFalse(isSuccess) + XCTAssertFalse(authManagerSpy.loginCalled) + XCTAssertNil(authManagerSpy.accessToken) + XCTAssertNil(authManagerSpy.refreshToken) + XCTAssertNil(authManagerSpy.loginType) + XCTAssertFalse(authManagerSpy.isLoggedIn == true) + } + + func test_애플_회원가입을_성공적으로_했을때_true를_반환하고_로그인_처리된다() async { + // arrange + guard let mockFileLocation = Bundle(for: type(of: self)).url(forResource: "LoginData", withExtension: "json"), + let mockData = try? Data(contentsOf: mockFileLocation) else { + XCTFail("Mock json 파일 로드 실패.") + return + } + + MockURLProtocol.requestHandler = { request in + let response = HTTPURLResponse(url: request.url!, + statusCode: 200, + httpVersion: nil, + headerFields: nil) + return (response, mockData, nil) + } + + // act + let isSuccess = await sut.signUp(withApple: "MockToken", username: "안유진") + + // assert + XCTAssertTrue(isSuccess) + XCTAssertTrue(authManagerSpy.loginCalled) + XCTAssertEqual(authManagerSpy.accessToken, "mockAccessToken") + XCTAssertEqual(authManagerSpy.refreshToken, "mockRefreshToken") + XCTAssertEqual(authManagerSpy.loginType, .apple) + XCTAssertTrue(authManagerSpy.isLoggedIn == true) + } + + func test_애플_회원가입을_실패했을때_false를_반환하고_로그인_처리되지_않는다() async { + // arrange + guard let mockFileLocation = Bundle(for: type(of: self)).url(forResource: "LoginData", withExtension: "json"), + let mockData = try? Data(contentsOf: mockFileLocation) else { + XCTFail("Mock json 파일 로드 실패.") + return + } + + MockURLProtocol.requestHandler = { request in + let response = HTTPURLResponse(url: request.url!, + statusCode: 400, + httpVersion: nil, + headerFields: nil) + return (response, mockData, nil) + } + + // act + let isSuccess = await sut.signUp(withApple: "MockToken", username: "안유진") + + // assert + XCTAssertFalse(isSuccess) + XCTAssertFalse(authManagerSpy.loginCalled) + XCTAssertNil(authManagerSpy.accessToken) + XCTAssertNil(authManagerSpy.refreshToken) + XCTAssertNil(authManagerSpy.loginType) + XCTAssertFalse(authManagerSpy.isLoggedIn == true) + } +} diff --git a/iOS/Layover/LayoverTests/Stubs/StubAuthManager.swift b/iOS/Layover/LayoverTests/Stubs/StubAuthManager.swift index 9558084..69eff83 100644 --- a/iOS/Layover/LayoverTests/Stubs/StubAuthManager.swift +++ b/iOS/Layover/LayoverTests/Stubs/StubAuthManager.swift @@ -8,13 +8,30 @@ @testable import Layover import Foundation -final class StubAuthManager: AuthManagerProtocol { +class StubAuthManager: AuthManagerProtocol { - // MARK: Properties + // MARK: - Properties var accessToken: String? = "Fake Access Token" var refreshToken: String? = "Fake Refresh Token" var isLoggedIn: Bool? = true var loginType: LoginType? = .kakao var memberID: Int? = -1 + + // MARK: - Methods + + func login(accessToken: String?, refreshToken: String?, loginType: LoginType?) { + self.accessToken = accessToken + self.refreshToken = refreshToken + self.loginType = loginType + isLoggedIn = true + } + + func logout() { + accessToken = nil + refreshToken = nil + memberID = nil + loginType = nil + isLoggedIn = false + } }