투케이2K

168. (swift5/xcode) [RxSwift] [Observable] retry 사용해 API 호출 재시도 요청 로직 처리 수행 - error count 본문

Swift

168. (swift5/xcode) [RxSwift] [Observable] retry 사용해 API 호출 재시도 요청 로직 처리 수행 - error count

투케이2K 2023. 11. 21. 17:03

[개발 환경 설정]

개발 툴 : XCODE

개발 언어 : SWIFT5

 

[소스 코드]

    // -----------------------------------------------------------------------------------------
    // MARK: - [테스트 메인 함수 정의 실시]
    // -----------------------------------------------------------------------------------------
    func testMain() {
        S_Log._D_(description: "테스트 함수 시작 실시", data: nil)
        
        
        /*
         ------------------------------------
         [요약 설명]
         ------------------------------------
         1. 반응형 프로그래밍 은 데이터의 흐름을 먼저 정의하고, 데이터가 변경되었을 때 [연관된 작업] 을 수행합니다
         ------------------------------------
         2. ReactiveX 는 Observer Pattern 옵저버 패턴 을 확장하며, Sequence 를 조합할 수 있는 연산자를 지원하며,
         low-level Thread, 동기화, Thread 안전성, non-blocking I/O에 관한 우려를 줄입니다
         ------------------------------------
         3. 필요 import :
         
         import RxSwift
         ------------------------------------
         4. 라이브러리 설치 참고 사이트 :
         
         https://blog.naver.com/kkh0977/223268744402
         https://github.com/ReactiveX/RxSwift
         ------------------------------------
         */
        
        
        // [로직 처리 실시]
        DispatchQueue.main.async {
            
            // [재시도 변수 선언]
            let maxRetryCount = 3
            
            // [Observable 생성]
            let subscription = self.observableCreate(data: "") // [input null send]
                .retry { error in
                    error.enumerated().flatMap { (tryCount, error) -> Observable<Int> in
                        if (tryCount + 1) > maxRetryCount {
                            
                            S_Log._D_(description: "observable :: retryWhen :: End", data: nil)
                            
                            return Observable.error(error)
                        }
                        
                        S_Log._D_(description: "observable :: retryWhen", data: ["\((tryCount + 1))"])
                        
                        return Observable<Int>.timer(.seconds(5), scheduler: MainScheduler.instance).take(1)
                        
                    }
                }
                .subscribe(
                    onNext: { element in // [발행 아이템 요소]
                        
                        S_Log._D_(description: "observable :: onNext", data: ["\(element)"])
                        
                })
                { onError in // [발행 에러]
                    
                    S_Log._D_(description: "observable :: onError", data: ["\(onError)"])
                    
                } onCompleted: { // [발행 완료]
                    
                    S_Log._D_(description: "observable :: onCompleted", data: nil)
                    
                } onDisposed: { // [구독 연결 및 해제]
                    
                    S_Log._D_(description: "observable :: onDisposed", data: nil)
                    
                }
            
        }

    }
    
    
    
    
    
    // -----------------------------------------------------------------------------------------
    // MARK: - [Observable.create 리턴 함수 선언]
    // -----------------------------------------------------------------------------------------
    func observableCreate(data: String) -> Observable<String> {
        
        return Observable<String>.create { observer in
            
            // [로직 처리 실시]
            DispatchQueue.main.async {
                
                // [인풋 값 널 체크]
                if data != "" {
                    
                    // [리턴 반환]
                    observer.onNext("\(data)")
                    observer.onCompleted()
                    
                }
                else {
                    
                    // [에러 반환]
                    observer.onError(NSError(domain: "observable.error", code: 0, userInfo: [NSLocalizedDescriptionKey: "Input Data Check Error"]))
                    observer.onCompleted()
                    
                }
                
            }
            return Disposables.create()
            
        }

    }
 

[결과 출력]


 

반응형
Comments