Forked provides a generalized approach to managing shared data in Swift applications, taking control of data races and race conditions. It can be used to replace — or be used in conjunction with — actors, locks, and queues.
Forked can operate within a single iOS app, on a Swift server, or distributed across a network. The ForkedCloudKit
package, for example, supports syncing of data across devices in just a few lines of code.
In short, what's forking stopping you?!1
Who wants to invest time in a framework without knowing if it's right for them? Nobody, that's who!
So we have uploaded the Forkers sample app to the App Store for you to try. (Note that it is unlisted, so use the link instead of searching.) The Forkers app is built on Forked, and the source code is right here. Try it out, and don't forget to test out the iCloud sync!
Add to your Package.swift
:
dependencies: [
.package(url: "https://github.com/drewmccormack/Forked.git", from: "0.1.0")
]
- Select your project in the navigator
- Open the Package Dependencies tab
- Click + and enter:
https://github.com/drewmccormack/Forked.git
- Add
Forked
and any of the subpackages you need
- Safe: Prevents data races and manages race conditions without locks, queues, or actors
- Swift Values: Model data with
Sendable
value types that pass easily between threads and isolation domains - Simple Setup: 100% Swift, no complex configuration needed
- Smart Sync: Git-inspired branching and merging lets you track changes on a single device or across many
- Supports Local-First: Local-first apps are all the rage, and Forked makes it easy to get started
- Smashable: Advanced 3-way merging algorithms (eg CRDTs) intelligently handle conflicts
- Saveable: Full
Codable
support for easy persistence to disk and cloud services - Seamless iCloud: Built-in CloudKit integration for effortless multi-device synchronization
- Scalable: You can start using Forked with your own data types, and scale up to complete data models when it suits
- Self Service: You can add custom storage for Forked data, and integrate with custom cloud services
- Succinct: Unlike Git, Forked only keeps the bare essentials for merging, not a complete history of all changes
Forked is based on a decentral model similar to Git. It tracks changes to a shared data resource, and resolves conflicts using 3-way merging. You are in control, and never lose any changes to your data.
In contrast to locks, queues, and actors, Forked doesn't serialize access to a resource. Instead, Forked provides a branching mechanism to systematically create copies of the data, which can be modified concurrently, and merged at a later time.
Forked takes care of all the logic involved in the branching process, including keeping a copy of the data at the point that branches (known as forks) diverge. This 'divergence' copy is known as the common ancestor, and it is important, because when it comes time to merge the forks again, Forked can use it to determine what was changed, and in which fork(s).
You can merge branches safely at any time with Forked — in any order — using powerful merging algorithms that go way beyond what is available in other data modeling frameworks. For example, Forked utilizes so-called Conflict-Free Replicated Data Types (CRDTs) to merge text in a way that would seem logical to people, rather than choosing a solution with results only a machine could love.
Ready to play? Let's learn about Forked by example.
Here is your first fork:
import Forked
let uiFork = Fork(name: "ui")
let intResource = QuickFork<Int>(initialValue: 0, forks: [uiFork])
QuickFork
is a convenient way to create an in-memory ForkedResource
holding a single value, in this case an Int
.
We have also declared uiFork
, which is a named fork. Aside from forks you create yourself, all ForkedResource
instances have a central fork called main
, which can be merged with any other fork.
Let's update the Int
on uiFork
, and independently on the main
fork, then merge to get the result:
try intResource.update(uiFork, with: 1)
try intResource.update(.main, with: 2)
try intResource.mergeIntoMain(from: uiFork)
let resultInt = try intResource.value(in: .main)!
The resultInt
will be 2
in this case, because that was the value set most recently.
For an atomic type like an Int
, the results of merging are not very interesting; the real power of Forked comes from its ability to merge complex data types.
Let's start by defining a struct so we can control the merging behavior of the Int
.
struct AccumulatingInt: Mergeable {
var value: Int = 0
func merged(withSubordinate other: Self, commonAncestor: Self) throws -> Self {
return AccumulatingInt(value: self.value + other.value - commonAncestor.value)
}
}
By conforming to the Mergeable
protocol, AccumulatingInt
has total control over how it is merged.
The Mergeable
protocol requires the func merged(withSubordinate:commonAncestor:)
. The subordinate
is a conflicting value from another fork, and commonAncestor
is the value at the point that the two forks diverged.
The merge algorithm of AccumulatingInt
determines what has changed on each fork since the common ancestor was created, and tallies these changes up to produce a new value.
If we were to use an AccumulatingInt
in the original example, instead of an Int
, the result would be 3
, because the uiFork
incremented by 1
, and the main
fork incremented by 2
, giving a total of 3
.
So you can come up with structs that can merge in any way that you choose, but those merging algorithms can quickly get complex. That's where the subpackage ForkedMerge
comes in: it provides standard built-in merging algorithms.
Imagine we are developing a text editor with this oversimplified model:
import Forked
import ForkedMerge
struct TextDocument: Mergeable {
var text: String = ""
func merged(withSubordinate other: Self, commonAncestor: Self) throws -> Self {
let newText = try TextMerger().merge(
self.text,
withSubordinate: other.text,
commonAncestor: commonAncestor.text
)
return TextDocument(text: newText)
}
}
It doesn't look like much, but you've just created the model for a fully collaborative text editor. For example, if the model initially contains the text "Fork Yeah", and...
- One user changes this to "Fork Yeah!!!"
- Another changes it at the same time to "Fork yeah"
TextMerger
will merge to give "Fork yeah!!!"
Having complete control over merging is great, and the merging algorithms provided by ForkedMerge
make it much easier to piece things together, but wouldn't it be nice if Forked
could just generate this code automatically?
That's exactly what ForkedModel
is for. It uses Swift Macros to make defining a global data model almost trivial.
Let's update TextDocument
to use ForkedModel
:
import Forked
import ForkedModel
@ForkedModel
struct TextDocument {
@Merged var text: String = ""
}
"Where's the rest?" I hear you cry. There is no rest! That's the forking lot!
This code is equivalent to the code we wrote manually in the previous section. It could form the basis of a fully collaborative text editor, or simply a personal editor syncing via iCloud.
And it doesn't stop there: each property in the struct gets merged independently. You can choose from a standard atomic merge for simple types, to advanced merging algorithms for common Swift types like String
, Array
, Dictionary
, and Set
.
To demonstrate, here is a more complex example of TextDocument
:
import Forked
import ForkedModel
@ForkedModel
struct TextDocument {
var id: UUID = UUID()
@Merged var text: String = ""
@Merged var tags: Set<String> = []
@Merged(using: .textMerge) var comment: String = ""
@Merged var editCount: AccumulatingInt = .init()
var cursorPosition: Int = 0
}
The @Merged
attribute tells ForkedModel
that the property is Mergeable
, and it should use an appropriate merging algorithm. There are defaults for most common types, but you can override this by passing a different merge algorithm to the using:
parameter.
If you have a custom Mergeable
type, like AccumulatingInt
, applying @Merged
will cause it to merge using the merged(withSubordinate:commonAncestor:)
method you provided.
Properties without @Merged
attached will be merged atomically, with a more recent change taking precedence over an older one. Properties will be merged in a property-wise manner, based on the most recent change to the property itself
A good way to get started with Forked
is to take a look at the sample apps provided. They range in difficulty from very basic, to a fully-functional iCloud-based Contacts app.
Actors solve the problem of data races in Swift very well, but they don't help at all with race conditions, and can even give rise to new ones. This sample shows you can use a ForkedResource
inside of an actor to deal with race conditions in a straightforward way.
Sets up a simple mergeable model similar to the ones above. The UI allows you to change the values of text and a counter in two different forks, and pressing a button you see how they get merged.
The model in this sample is extremely simple, and is secondary in importance to how you setup the CloudKitExchange
to sync data with iCloud. The sample shows how you can use a ForkedResource
for storage on disk, update a property for display in SwiftUI, and monitor changes to forks in order to refresh the UI when changes arrive from iCloud.
Forkers is a contacts app for keeping track of your favorite forkers. The model is more complex than the other samples, showing how you can nest Mergeable
types, in this case with an Array
of your contacts. It also integrates with iCloud, giving a fully-functional, local-first contacts app.
Documentation is available for each subpackage.
This is the core package, and needed to use any of the other packages. It provides ForkedResource
, which is the basic building block of Forked
.
This package provides the standard merging algorithms for Mergeable
types. It also includes a number of Conflict-Free Replicated Data Types (CRDTs).
This package provides the @ForkedModel
and @Merged
macros, which allow you to define a global data model using value types.
This provides the CloudKitExchange
class, which automatically syncs a ForkedResource
between devices with iCloud.
Contributions are welcome! Please feel free to submit a pull request or open an issue.
Forked is available under the MIT license. See the LICENCE file for more info.
Footnotes
-
"Forking" jokes are inspired by The Good Place. Go watch it! ↩