/SRNetworkLayer

A robust Swift-based network layer for iOS applications, supporting Combine and async/await for seamless asynchronous operations.

Primary LanguageSwift

SRNetworkLayer

SRNetworkLayer is a Swift-based network layer designed to simplify network communication in your iOS applications. It provides a robust and flexible structure for making network requests, handling responses, and managing dependencies.

Features

  • Asynchronous network requests using Combine and async/await.
  • Support for file uploads with progress tracking.
  • Retry logic with customizable interceptors.
  • Dependency injection for better testability and modularity.
  • Comprehensive error handling and logging.

Requirements

  • iOS 13.0+
  • Swift 5.0+
  • Xcode 11.0+

Usage

Setting Up the Environment

Initialize the AppEnvironment in your AppDelegate or SceneDelegate:

@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {

    var window: UIWindow?

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
        let _ = AppEnvironment.setup()
        return true
    }
}

Making a Network Request

To make a network request, use the NetworkRepositories class:

import Combine
import Foundation

// MARK: - LoginViewModel

class LoginViewModel: BaseViewModel<GeneralErrorResponse> {
    @Published var userModel: UserResponseModel?
    @Published var isLoading: Bool = false
    var loginCancellableSet = Set<AnyCancellable>()
}

extension LoginViewModel {
    func login(email: String, password: String) {
        isLoading = true
        remoteRepositories.loginServices?
            .login(email: email, password: password)
            .sink(receiveCompletion: { [weak self] completion in
                switch completion {
                case .finished:
                    break
                case .failure(let failure):
                    self?.isLoading = false
                    self?.error.send(failure)
                }
            }, receiveValue: { [weak self] model in
                self?.isLoading = false
                self?.userModel = model
            }).store(in: &loginCancellableSet)
    }

    @MainActor
    func asyncLogin(email: String, password: String) {
        isLoading = true
        Task {
            do {
                let response = try await remoteRepositories.loginServices?.asyncLogin(email: email, password: password)
                userModel = response
                isLoading = false
            } catch let error as NetworkError<GeneralErrorResponse> {
                isLoading = false
                self.error.send(error)
            }
        }
    }
}

Handling File Uploads

To upload a file, use the asyncUploadRequest method:

class UploadViewModel: BaseViewModel<GeneralErrorResponse> {

    @MainActor
    func uploadProfilePicture(file: Data?, fileName: String) {
        isLoading = true
        Task {
            do{
                let response = try await remoteRepositories.profilePictureServices?.asyncUpload(file: file, name: fileName, uploadProgress: { [weak self] progress in
                    guard let _ = self else {return}
                    debugPrint(progress)
                })
                self.isLoading = false
                self.uploadPhotoModel = response
            }catch let error as NetworkError{
                self.isLoading = false
                self.error.send(error)
            }
        }
    }
}

Documentation

For full documentation, please refer to the source code and the comments within. Each method is documented to provide clarity on its purpose, parameters, return values, and potential errors.

License

SRNetworkLayer is released under the MIT license. See LICENSE for details.