From 5103591d57ebd649b05bf931e2be0d0fc9fc200f Mon Sep 17 00:00:00 2001 From: Levi Bostian Date: Sat, 4 May 2024 06:50:49 -0500 Subject: [PATCH] refactor!: delete non-async versions of functions As part of the efforts to improve the use of swift concurrency in the codebase https://github.com/levibostian/Wendy-iOS/issues/137, this commit changes the public API to use async/await functions only. Also, delete deprecated functions. Since the previous commit we made is also a breaking change, I decided it would be a good time to introduce more changes into this release. commit-id:bc534a74 --- README.md | 72 +++++++++---------- Source/Wendy.swift | 41 ++--------- .../PerformanceIntegrationTests.swift | 20 +++--- Tests/Integration/WendyIntegrationTests.swift | 30 +------- 4 files changed, 50 insertions(+), 113 deletions(-) diff --git a/README.md b/README.md index b01a43a..d2dc369 100644 --- a/README.md +++ b/README.md @@ -15,7 +15,7 @@ Android developer? [Check out the Android version of Wendy!][4] Wendy is under active development with lots of changes happening. **Wendy is usable today and ready to install in your app**. No matter if you're using Wendy already or not, expect... -* Frequent major version bumps (breaking changes) that may require code migrations in your code base. Luckily, each major version release contains up-to-date [migration documentation](https://github.com/levibostian/Wendy-iOS/blob/main/MIGRATION.md). +* Frequent major version bumps (breaking changes) that may require code migrations in your code base. Luckily, each major version release contains up-to-date [migration documentation][5]. * Best practices documentation out-of-date, but planned. The public API is kept up-to-date (this README) but not much guidance on suggested ways to use the API. The main focus at the moment is building and shipping major improvements to Wendy to give it the core feature-set to make it a no-brainer to use in an app. Then, the focus will transition over to less frequent major version bumps, best practice documentation, etc. @@ -24,9 +24,9 @@ This project holds a special place in my heart. I've enjoyed working on it for y My favorite part about writing code is hearing how others experience what I build. Send a message if you find Wendy interesting. I would love to hear it! -- Levi +- Levi -See the [latest announcement][5] to stay updated on the latest set of changes coming to Wendy. +See the [latest announcement][6] to stay updated on the latest set of changes coming to Wendy. ## What is Wendy? @@ -48,13 +48,13 @@ Wendy currently has the following functionality: # Install -Wendy-iOS is available through [CocoaPods][6]. To install it, simply add the following line to your Podfile: +Wendy-iOS is available through [CocoaPods][7]. To install it, simply add the following line to your Podfile: ```ruby pod 'Wendy', '~> version-here' ``` -(replace `version-here` with [![Version][image-6]][7]) +(replace `version-here` with [![Version][image-6]][8]) # Getting started @@ -89,12 +89,6 @@ To finish initialization, we need to create a task runner subclass. Create a new import Wendy class MyWendyTaskRunner: WendyTaskRunner { - func runTask(tag: String, data: Data?, complete: @escaping (Error?) -> Void) { - } -} - -// Or, use the Swift Concurrency version: -class MyWendyTaskRunner: WendyTaskRunnerConcurrency { func runTask(tag: String, data: Data?) async throws { } } @@ -109,8 +103,8 @@ Now that the SDK is initialized, it’s time to sync some data to our network AP In our Grocery List app example, we want to allow users to create new grocery items. Every time that a user creates a new grocery list item, we don't want to show them a progress bar saying, "Saving grocery list item..." while we perform an API call. We want to be able to *instantly* save that grocery list item and sync it with the cloud storage later so our user can get on with their life (can't you just see your App Store reviews going up, up, up right now? ⭐⭐⭐⭐⭐). There are 2 steps to setting up Wendy for syncing data. -1. [Adding tasks to Wendy][8] -2. [Writing the network code to perform the sync][9] +1. [Adding tasks to Wendy][9] +2. [Writing the network code to perform the sync][10] Let’s get into each of these steps. @@ -143,7 +137,7 @@ Let’s look at some example code that runs the grocery store list item. ```swift import Wendy -class MyWendyTaskRunner: WendyTaskRunnerConcurrency { +class MyWendyTaskRunner: WendyTaskRunner { func runTask(tag: String, data: Data?) async throws { switch tag { case "AddGroceryListItem": @@ -211,7 +205,7 @@ Besides listening for status changes of individual tasks, you can also listen to WendyConfig.addTaskRunnerListener(listener: listener) ``` -It’s suggested to view the [Best practices doc][10] to learn more about making a great experience in your offline-first app. +It’s suggested to view the [Best practices doc][11] to learn more about making a great experience in your offline-first app. ## Clear data @@ -247,10 +241,6 @@ Coming soon! You may be able to do this already, but it has not been tested. A good place to start would be clear Wendy before each test and use it like normal. See where that takes you. Report issues as you encounter them. -## Example - -To run the example project, clone the repo, and run `pod install` from the `Example/` directory first. Then, open XCode and run the project. - ## Documentation Wendy currently *does not* have full code documentation. It is planned to have full documentation generated via jazzy in the near future. @@ -258,7 +248,7 @@ Wendy currently *does not* have full code documentation. It is planned to have f Until then, the best thing to do is: * Read this README on how to get started. -* Wendy-Android has [full documentation created for it][11]. If you are wondering how a specific function works, you may be able to learn there. *Warning: Wendy-Android and Wendy-iOS are kept up to date between one another as soon as possible. When a bug is fixed on one, the other gets the same bug fixed on it as well. However, it may take a day or two for this sync to happen by the contributors. With that in mind, the documentation might be a tad bit off between the libraries.* +* Wendy-Android has [full documentation created for it][12]. If you are wondering how a specific function works, you may be able to learn there. *Warning: Wendy-Android and Wendy-iOS are kept up to date between one another as soon as possible. When a bug is fixed on one, the other gets the same bug fixed on it as well. However, it may take a day or two for this sync to happen by the contributors. With that in mind, the documentation might be a tad bit off between the libraries.* ## Configure Wendy @@ -292,7 +282,7 @@ WendyConfig.debug = true ## Maintainers -* Levi Bostian - [GitHub][12] +* Levi Bostian - [GitHub][13] ![Levi Bostian image][image-7] @@ -304,17 +294,17 @@ Wendy-iOS is available under the MIT license. See the LICENSE file for more info Wendy is open for pull requests. -**Want to add features to Wendy?** Before you decide to take a bunch of time and add functionality to the library, please, [create an issue][13] stating what you wish to add. This might save you some time in case your purpose does not fit well in the use cases of Wendy. +**Want to add features to Wendy?** Before you decide to take a bunch of time and add functionality to the library, please, [create an issue][14] stating what you wish to add. This might save you some time in case your purpose does not fit well in the use cases of Wendy. Follow the steps below to compile the Wendy project on your machine for contributing! * Install these development tools: -[mint][14] used to run other development CLI commands such as linter/formatter. -[taskfile][15] an alternative to `Makefile` to run commands. -[lefthook](https://github.com/evilmartians/lefthook/blob/HEAD/docs/install.md) for git hooks. +[mint][15] used to run other development CLI commands such as linter/formatter. +[taskfile][16] an alternative to `Makefile` to run commands. +[lefthook][17] for git hooks. -* Install git hooks: +* Install git hooks: `lefthook install` @@ -326,25 +316,27 @@ Follow the steps below to compile the Wendy project on your machine for contribu # Credits -Header photo by [Allef Vinicius][16] on [Unsplash][17] +Header photo by [Allef Vinicius][18] on [Unsplash][19] [1]: http://cocoapods.org/pods/Wendy [2]: http://cocoapods.org/pods/Wendy [3]: http://cocoapods.org/pods/Wendy [4]: https://github.com/levibostian/wendy-android -[5]: https://github.com/levibostian/Wendy-iOS/discussions/categories/announcements -[6]: http://cocoapods.org -[7]: http://cocoapods.org/pods/Wendy -[8]: #adding-tasks-to-wendy -[9]: #writing-the-network-code-to-perform-the-sync -[10]: BEST_PRACTICES.md -[11]: https://levibostian.github.io/Wendy-Android/wendy/ -[12]: https://github.com/levibostian -[13]: https://github.com/levibostian/Wendy-iOS/issues/new -[14]: https://github.com/yonaskolb/Mint?tab=readme-ov-file#installing -[15]: https://taskfile.dev/installation/ -[16]: https://unsplash.com/photos/FPDGV38N2mo?utm_source=unsplash&utm_medium=referral&utm_content=creditCopyText -[17]: https://unsplash.com/search/photos/red-head?utm_source=unsplash&utm_medium=referral&utm_content=creditCopyText +[5]: https://github.com/levibostian/Wendy-iOS/blob/main/MIGRATION.md +[6]: https://github.com/levibostian/Wendy-iOS/discussions/categories/announcements +[7]: http://cocoapods.org +[8]: http://cocoapods.org/pods/Wendy +[9]: #adding-tasks-to-wendy +[10]: #writing-the-network-code-to-perform-the-sync +[11]: BEST_PRACTICES.md +[12]: https://levibostian.github.io/Wendy-Android/wendy/ +[13]: https://github.com/levibostian +[14]: https://github.com/levibostian/Wendy-iOS/issues/new +[15]: https://github.com/yonaskolb/Mint?tab=readme-ov-file#installing +[16]: https://taskfile.dev/installation/ +[17]: https://github.com/evilmartians/lefthook/blob/HEAD/docs/install.md +[18]: https://unsplash.com/photos/FPDGV38N2mo?utm_source=unsplash&utm_medium=referral&utm_content=creditCopyText +[19]: https://unsplash.com/search/photos/red-head?utm_source=unsplash&utm_medium=referral&utm_content=creditCopyText [image-1]: https://img.shields.io/cocoapods/v/Wendy.svg?style=flat [image-2]: https://img.shields.io/cocoapods/l/Wendy.svg?style=flat diff --git a/Source/Wendy.swift b/Source/Wendy.swift index cf3b4c3..35fde06 100644 --- a/Source/Wendy.swift +++ b/Source/Wendy.swift @@ -97,7 +97,9 @@ public final class Wendy: Sendable, Singleton { } LogUtil.d("Wendy is configured to automatically run tasks. Wendy will now attempt to run newly added task: \(task.describe())") - runTask(task.taskId!, onComplete: nil) + Task { + _ = await runTask(task.taskId!) + } return true } @@ -112,14 +114,6 @@ public final class Wendy: Sendable, Singleton { return result } - public func runTask(_ taskId: Double, onComplete: (@Sendable (TaskRunResult) -> Void)?) { - Task { - let result = await self.runTask(taskId) - - onComplete?(result) - } - } - @discardableResult public func runTasks(filter: RunAllTasksFilter? = nil) async -> PendingTasksRunnerResult { let result = await pendingTasksRunner.runAllTasks(filter: filter) @@ -127,14 +121,6 @@ public final class Wendy: Sendable, Singleton { return result } - public func runTasks(filter: RunAllTasksFilter? = nil, onComplete: (@Sendable (PendingTasksRunnerResult) -> Void)?) { - Task { - let result = await self.runTasks(filter: filter) - - onComplete?(result) - } - } - public final func getAllTasks() -> [PendingTask] { DIGraph.shared.pendingTasksManager.getAllTasks() } @@ -144,15 +130,15 @@ public final class Wendy: Sendable, Singleton { Note: If a task is currently running when clear() is called, that running task will be finish executing but will not run again in the future as it has been cancelled. */ - public final func clear() { + public final func clear() async { /// It's not possible to stop a dispatch queue of tasks so there is no way to stop the currently running task runner. /// This solution of using UserDefaults to set a threshold solves that problem while also leaving Wendy untouched to continue running as usual. If we deleted all data, as Android's Wendy does, we would have potential issues with tasks that are still in the queue but core data and userdefaults being deleted causing potential crashes and IDs being misaligned. PendingTasksUtil.setValidPendingTasksIdThreshold() LogUtil.d("Wendy tasks set as cancelled. Currently scheduled Wendy tasks will all skip running.") // Run all tasks (including manually run tasks) as they are all cancelled so it allows them all to be cleared fro the queue now and listeners can get notified. - for task in getAllTasks() { - runTask(task.taskId!, onComplete: nil) - } + + // We want to make it so the next time the developer calls runTasks(), 0 tasks get run. To do that, you must run the queue to have all tasks deleted after they are marked as invalid. + await runTasks() } public func addQueueReader(_ reader: QueueReader) { @@ -168,19 +154,6 @@ public final class Wendy: Sendable, Singleton { } } -// Public API functions for backwards compatibility. -public extension Wendy { - @available(*, deprecated, message: "Use addTask(tag:data:groupId:) instead.") - func addTask(tag: String, dataId: String?, groupId: String? = nil) -> Double { - addTask(tag: tag, data: dataId, groupId: groupId) - } - - @available(*, deprecated, message: "Use addTask(tag:data:groupId:) instead.") - func addTask(tag: Tag, dataId: String?, groupId: String? = nil) -> Double where Tag.RawValue == String { - addTask(tag: tag.rawValue, data: dataId, groupId: groupId) - } -} - extension DIGraph { var taskRunner: WendyTaskRunner? { Wendy.shared.taskRunner diff --git a/Tests/Integration/PerformanceIntegrationTests.swift b/Tests/Integration/PerformanceIntegrationTests.swift index df3e950..cf99588 100644 --- a/Tests/Integration/PerformanceIntegrationTests.swift +++ b/Tests/Integration/PerformanceIntegrationTests.swift @@ -56,7 +56,8 @@ class PerformanceIntegrationTests: TestClass { // When we begin running all tasks, ask Wendy to run task 3, which means it would run it before task 2. if tagOfTaskWeAreRunning == "task1" { - Wendy.shared.runTask(3) { _ in + Task { + _ = await Wendy.shared.runTask(3) expectToFinishRunningSingleTask.fulfill() } } @@ -70,9 +71,8 @@ class PerformanceIntegrationTests: TestClass { } } - Wendy.shared.runTasks { _ in - expectToFinishRuningAllTasks.fulfill() - } + await Wendy.shared.runTasks() + expectToFinishRuningAllTasks.fulfill() await fulfillment(of: [ expectToRunTask1, @@ -83,7 +83,7 @@ class PerformanceIntegrationTests: TestClass { ], timeout: 1.0, enforceOrder: true) } - func test_runAllTasks_givenAlreadyRunning_expectIgnoreRequest() { + func test_runAllTasks_givenAlreadyRunning_expectIgnoreRequest() async { let expectToFinishRunningAllTasks = expectation(description: "expect to finish running all tasks") let expectToIgnoreRequestToRunAllTasks = expectation(description: "expect to ignore request to run all tasks") @@ -93,17 +93,17 @@ class PerformanceIntegrationTests: TestClass { taskRunnerStub.runTaskClosure = { tagOfTaskWeAreRunning, _ in // When we begin running all tasks, ask Wendy to run all tasks again. The request should be ignored so it should complete fast. if tagOfTaskWeAreRunning == "task1" { - Wendy.shared.runTasks { _ in + Task { + _ = await Wendy.shared.runTasks() expectToIgnoreRequestToRunAllTasks.fulfill() } } } - Wendy.shared.runTasks { _ in - expectToFinishRunningAllTasks.fulfill() - } + await Wendy.shared.runTasks() + expectToFinishRunningAllTasks.fulfill() - wait(for: [ + await fulfillment(of: [ expectToIgnoreRequestToRunAllTasks, expectToFinishRunningAllTasks ], timeout: 1.0, enforceOrder: true) diff --git a/Tests/Integration/WendyIntegrationTests.swift b/Tests/Integration/WendyIntegrationTests.swift index 460ee33..b7e67cb 100644 --- a/Tests/Integration/WendyIntegrationTests.swift +++ b/Tests/Integration/WendyIntegrationTests.swift @@ -27,33 +27,6 @@ class WendyIntegrationTests: TestClass { XCTAssertNotEqual(taskGroup1, taskGroup2) } - func test_addTasks_givenDeprecatedDataId_expectToAddAndRunTaskUsingDataId() async { - enum Tag: String { - case foo - } - - Wendy.shared.addTask(tag: "string-tag", dataId: "string-dataId") - Wendy.shared.addTask(tag: Tag.foo, dataId: "enum-dataId") - - taskRunnerStub.runTaskClosure = { tag, data in - guard let dataId: String = data?.wendyDecode() else { - XCTFail("Could not decode data") - return - } - - switch tag { - case "string-tag": - XCTAssertEqual(dataId, "string-dataId") - case Tag.foo.rawValue: - XCTAssertEqual(dataId, "enum-dataId") - default: - XCTFail("Unexpected tag") - } - } - - _ = await runAllTasks() - } - func test_addTasks_expectAddTasksAndRunTasksGivenEnumInsteadOfStrings() async { enum AsyncTasks: String { case foo @@ -298,9 +271,8 @@ class WendyIntegrationTests: TestClass { Wendy.shared.addTask(tag: "tag", data: "dataId") Wendy.shared.addTask(tag: "tag", data: "dataId") - Wendy.shared.clear() + await Wendy.shared.clear() - sleep(1) // give wendy time to run all scheduled tasks that do the deleting. let runTasksResults = await runAllTasks() XCTAssertEqual(runTasksResults.numberTasksRun, 0) }