Skip to content

A generic polling mechanism written with Swift Concurrency and Swift 6 standard

License

Notifications You must be signed in to change notification settings

SergeyPetrachkov/AsyncPoller

Repository files navigation

AsyncPoller

AsyncPoller is a Swift package that provides a simple, actor-based utility for performing asynchronous polling operations. It allows you to periodically execute a task until a certain condition is met or a timeout occurs.

Features

  • Configurable polling interval and timeout.
  • Asynchronous and thread-safe execution using Swift's concurrency model.
  • Customizable completion condition to determine when the polling should stop.

Installation

Swift Package Manager

You can add AsyncPoller to your project using Swift Package Manager. In your Package.swift file, add the following dependency:

dependencies: [
    .package(url: "https://github.com/SergeyPetrachkov/AsyncPoller.git", from: "1.0.0")
]

Then, include "AsyncPoller" as a dependency in your target:

targets: [
    .target(
        name: "YourTarget",
        dependencies: ["AsyncPoller"]
    )
]

Usage

To use AsyncPoller, first import the module:

import AsyncPoller

Creating a Poller

Create a new instance of AsyncPoller by providing a configuration, a completion condition, and a polling job.

let poller = AsyncPoller<String>(
    configuration: SimplePollingConfiguration(pollingInterval: 5, timeoutInterval: 60),
    completionCondition: { result in
        return result == "success"
    },
    pollingJob: {
        // Your async job that returns a result of type T (e.g., String in this case)
        return await fetchStatusFromServer()
    }
)

Starting the Poller

Start the polling process by calling the start() method. This method will run your polling job at the specified interval until the completion condition is met or the timeout occurs.

Task {
    do {
        let result = try await poller.start()
        print("Polling succeeded with result: \(result)")
    } catch {
        print("Polling failed with error: \(error)")
    }
}

Configuration

AsyncPoller is configured using the PollingConfigurating protocol, which specifies the polling interval and the timeout interval.

The default is this:

public struct SimplePollingConfiguration: PollingConfigurating {
    public let pollingInterval: TimeInterval
    public let timeoutInterval: TimeInterval

    public init(pollingInterval: TimeInterval, timeoutInterval: TimeInterval) {
        self.pollingInterval = pollingInterval
        self.timeoutInterval = timeoutInterval
    }

    public func pollingInterval(iteration: Int) -> TimeInterval {
        pollingInterval
    }
}

An example of a Polling configuration that uses Fibonacci numbers to determine the polling interval:

struct FibonacciPollingConfiguration: PollingConfigurating {
    let timeoutInterval: TimeInterval
    func pollingInterval(iteration: Int) -> TimeInterval {
        guard iteration > 1 else {
            return TimeInterval(iteration)
        }

        var dp = [Int](repeating: 0, count: iteration + 1)
        dp[1] = 1

        for i in 2...iteration {
            dp[i] = dp[i - 1] + dp[i - 2]
        }

        return TimeInterval(dp[iteration])
    }
}

Errors

AsyncPoller can throw the following errors:

PollingError.timeout: Thrown when the polling operation exceeds the specified timeout interval. PollingError.alreadyPolling: Thrown when an attempt is made to start a new polling operation while another one is already running.

Example

Here's a complete example demonstrating how to use AsyncPoller:

import Foundation
import AsyncPoller

func fetchStatusFromServer() async -> String {
    // Simulate an async job
    return "success"
}

let poller = AsyncPoller<String>(
    configuration: .init(pollingInterval: 5, timeoutInterval: 60),
    completionCondition: { result in
        return result == "success"
    },
    pollingJob: {
        return await fetchStatusFromServer()
    }
)

Task {
    do {
        let result = try await poller.start()
        print("Polling succeeded with result: \(result)")
    } catch {
        print("Polling failed with error: \(error)")
    }
}

License

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

Contributing

Contributions are welcome! Please open an issue or submit a pull request for any changes.

Contact

For any questions or feedback, please contact petrachkovsergey@gmail.com

About

A generic polling mechanism written with Swift Concurrency and Swift 6 standard

Topics

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages