swift-concurrency-developer
Swift Concurrency Developer (Smart Router)
Purpose
Expert guidance on Swift's concurrency system using the "Office Building" mental model from Fucking Approachable Swift Concurrency, combined with comprehensive reference material from Swift Concurrency Course.
When Auto-Activated
- Working with actors, isolation, Sendable, TaskGroups
- Keywords:
actor,isolation,Sendable,TaskGroup,nonisolated,async let - Fixing concurrency warnings or data race issues
Agent Behavior Contract (Follow These Rules)
- Analyze the project/package file to find out which Swift language mode (Swift 5.x vs Swift 6) and which Xcode/Swift toolchain is used when advice depends on it.
- Before proposing fixes, identify the isolation boundary:
@MainActor, custom actor, actor instance isolation, or nonisolated. - Do not recommend
@MainActoras a blanket fix. Justify why main-actor isolation is correct for the code. - Prefer structured concurrency (child tasks, task groups) over unstructured tasks. Use
Task.detachedonly with a clear reason. - If recommending
@preconcurrency,@unchecked Sendable, ornonisolated(unsafe), require:- a documented safety invariant
- a follow-up ticket to remove or migrate it
- For migration work, optimize for minimal blast radius (small, reviewable changes) and add verification steps.
- Course references are for deeper learning only. Use them sparingly and only when they clearly help answer the developer's question.
Project Settings Discovery
When analyzing Swift projects for concurrency issues:
-
Project Settings Discovery
- Use
ReadonPackage.swiftfor SwiftPM settings (tools version, strict concurrency flags, upcoming features) - Use
GrepforSWIFT_STRICT_CONCURRENCYorSWIFT_DEFAULT_ACTOR_ISOLATIONin.pbxprojfiles
- Use
-
Manual checks
- SwiftPM: Check
Package.swiftfor.enableExperimentalFeature("StrictConcurrency=targeted")or similar - Xcode projects: Search
project.pbxprojforSWIFT_DEFAULT_ACTOR_ISOLATION,SWIFT_STRICT_CONCURRENCY
- SwiftPM: Check
Core Mental Model: The Office Building
Think of your app as an office building where isolation domains are private offices with locks:
| Concept | Office Analogy | Swift |
|---|---|---|
| MainActor | Front desk (handles all UI) | @MainActor |
| actor | Department offices (Accounting, Legal) | actor BankAccount { } |
| nonisolated | Hallways (shared space) | nonisolated func name() |
| Sendable | Photocopies (safe to share) | struct User: Sendable |
| Non-Sendable | Original documents (stay in one office) | class Counter { } |
Key insight: You can't barge into someone's office. You knock (await) and wait.
Quick Decision Tree
When a developer needs concurrency guidance:
-
Starting fresh with async code?
- Read
references/async-await-basics.mdfor foundational patterns - For parallel operations →
references/tasks.md(async let, task groups)
- Read
-
Protecting shared mutable state?
- Need to protect class-based state →
references/actors.md(actors, @MainActor) - Need thread-safe value passing →
references/sendable.md(Sendable conformance)
- Need to protect class-based state →
-
Managing async operations?
- Structured async work →
references/tasks.md(Task, child tasks, cancellation) - Streaming data →
references/async-sequences.md(AsyncSequence, AsyncStream)
- Structured async work →
-
Working with legacy frameworks?
- Core Data integration →
references/core-data.md - General migration →
references/migration.md
- Core Data integration →
-
Performance or debugging issues?
- Slow async code →
references/performance.md(profiling, suspension points) - Testing concerns →
references/testing.md(XCTest, Swift Testing)
- Slow async code →
-
Understanding threading behavior?
- Read
references/threading.mdfor thread/task relationship and isolation
- Read
-
Memory issues with tasks?
- Read
references/memory-management.mdfor retain cycle prevention
- Read
Triage-First Playbook (Common Errors -> Next Best Move)
- SwiftLint concurrency-related warnings
- Use
references/linting.mdfor rule intent and preferred fixes; avoid dummy awaits as "fixes".
- Use
- "Sending value of non-Sendable type ... risks causing data races"
- First: identify where the value crosses an isolation boundary
- Then: use
references/sendable.mdandreferences/threading.md
- "Main actor-isolated ... cannot be used from a nonisolated context"
- First: decide if it truly belongs on
@MainActor - Then: use
references/actors.md(global actors,nonisolated, isolated parameters)
- First: decide if it truly belongs on
- XCTest async errors like "wait(...) is unavailable from asynchronous contexts"
- Use
references/testing.md(await fulfillment(of:)and Swift Testing patterns)
- Use
- Core Data concurrency warnings/errors
- Use
references/core-data.md(DAO/NSManagedObjectID, default isolation conflicts)
- Use
Quick Patterns
Async/Await
func fetchUser(id: Int) async throws -> User {
let (data, _) = try await URLSession.shared.data(from: url)
return try JSONDecoder().decode(User.self, from: data)
}
Parallel Work with async let
async let avatar = fetchImage("avatar.jpg")
async let banner = fetchImage("banner.jpg")
return Profile(avatar: try await avatar, banner: try await banner)
Tasks
// SwiftUI - cancels when view disappears
.task { avatar = await downloadAvatar() }
// Manual task (inherits actor context)
Task { await saveProfile() }
TaskGroup for Dynamic Parallel Work
try await withThrowingTaskGroup(of: Void.self) { group in
group.addTask { avatar = try await downloadAvatar() }
group.addTask { bio = try await fetchBio() }
try await group.waitForAll()
}
Actors
actor BankAccount {
var balance: Double = 0
func deposit(_ amount: Double) { balance += amount }
// No await needed - can access directly inside actor
nonisolated func bankName() -> String { "Acme Bank" }
}
await account.deposit(100) // Must await from outside
let name = account.bankName() // No await needed
Sendable Types
// Automatically Sendable - value type
struct User: Sendable {
let id: Int
let name: String
}
// Thread-safe class with internal synchronization
final class ThreadSafeCache: @unchecked Sendable {
private let lock = NSLock()
private var storage: [String: Data] = [:]
}
Common Mistakes
1. Thinking async = background
// WRONG: Still blocks main thread!
@MainActor func slowFunction() async {
let result = expensiveCalculation() // Synchronous = blocking
}
// CORRECT: Use detached task for CPU-heavy work
Task.detached(priority: .userInitiated) {
let result = expensiveCalculation()
await MainActor.run { updateUI(result) }
}
Production impact: Apps get rejected for "became unresponsive." See
references/production-pitfalls.mdsection 2.
2. Creating too many actors
Most things can live on MainActor. Only create actors when you have shared mutable state that can't be on MainActor.
3. Using MainActor.run unnecessarily
// WRONG
await MainActor.run { self.data = data }
// CORRECT - annotate the function
@MainActor func loadData() async { self.data = await fetchData() }
4. Blocking the cooperative thread pool (violates runtime contract)
Never use DispatchSemaphore, DispatchGroup.wait(), or condition variables in async code.
Why: These primitives hide dependencies from the runtime. The cooperative thread pool has a contract that threads will always make forward progress. Blocking primitives violate this contract and can cause deadlock.
// ❌ DANGEROUS: Can deadlock the cooperative pool
let semaphore = DispatchSemaphore(value: 0)
Task {
await doWork()
semaphore.signal()
}
semaphore.wait() // Thread blocked, runtime unaware
// ✅ Use async/await instead
let result = await doWork()
Debug tip: Set LIBDISPATCH_COOPERATIVE_POOL_STRICT=1 to catch blocking calls during development.
5. Creating unnecessary Tasks
// WRONG - unstructured
Task { await fetchUsers() }
Task { await fetchPosts() }
// CORRECT - structured concurrency
async let users = fetchUsers()
async let posts = fetchPosts()
await (users, posts)
6. Making everything Sendable
Not everything needs to cross boundaries. Ask if data actually moves between isolation domains.
7. Not batching MainActor hops
The main thread is separate from the cooperative thread pool. Each hop to/from MainActor requires a full context switch.
// ❌ Multiple context switches
for item in items {
let processed = await processItem(item)
await MainActor.run { displayItem(processed) } // Context switch per item
}
// ✅ Single context switch
let processed = await processAllItems(items)
await MainActor.run {
for item in processed { displayItem(item) }
}
8. Async for loops silently losing data
Using try? or empty catch {} in async loops swallows failures. Users lose data with zero indication. Acceptable for fire-and-forget (cache warming, analytics), dangerous for uploads/sync/migration. See references/production-pitfalls.md section 1.
9. Ignoring Task cancellation in long-running loops
for await under .task modifier is safe (structured concurrency propagates cancellation). But for await or while loops in stored Task { } properties need explicit Task.isCancelled checks. See references/production-pitfalls.md section 3.
10. Manual migration pitfalls (@preconcurrency + DispatchQueue mixing)
@preconcurrency and nonisolated(unsafe) hide real data races. Mixing DispatchQueue with async/await creates confusing execution contexts. Always document safety invariants and plan removal. See references/production-pitfalls.md section 4.
11. Task inside onAppear instead of .task modifier
Task { } in .onAppear is unstructured: not cancelled on disappear, fires on every re-appear. Use .task { } for async work tied to view lifecycle, .onAppear for sync-only setup. See references/production-pitfalls.md section 5.
Quick Reference
| Keyword | Purpose |
|---|---|
async |
Function can pause |
await |
Pause here until done |
Task { } |
Start async work, inherits context |
Task.detached { } |
Start async work, no context |
@MainActor |
Runs on main thread |
actor |
Type with isolated mutable state |
nonisolated |
Opts out of actor isolation |
Sendable |
Safe to pass between isolation domains |
@unchecked Sendable |
Trust me, it's thread-safe |
async let |
Start parallel work |
TaskGroup |
Dynamic parallel work |
When the Compiler Complains
Trace the isolation: Where did it come from? Where is code trying to run? What data crosses a boundary?
The answer is usually obvious once you ask the right question.
Reference Files
Load these files as needed for specific topics:
Foundational Concepts
async-await-basics.md- async/await syntax, execution order, async let, URLSession patternstasks.md- Task lifecycle, cancellation, priorities, task groups, structured vs unstructuredthreading.md- Thread/task relationship, suspension points, isolation domains, nonisolatedglossary.md- Quick definitions of core concurrency terms
Isolation & Safety
actors.md- Actor isolation, @MainActor, global actors, reentrancy, custom executors, Mutexsendable.md- Sendable conformance, value/reference types, @unchecked, region isolationmemory-management.md- Retain cycles in tasks, memory safety patterns
Advanced Patterns
async-sequences.md- AsyncSequence, AsyncStream, when to use vs regular async methodsasync-algorithms.md- Swift Async Algorithms package, combining sequencestask-local-values.md- Task-local context propagation, tracing, logging patternscore-data.md- NSManagedObject sendability, custom executors, isolation conflicts
Quality & Migration
performance.md- Profiling with Instruments, reducing suspension points, execution strategiestesting.md- XCTest async patterns, Swift Testing, concurrency testing utilitiesmigration.md- Swift 6 migration strategy, closure-to-async conversion, @preconcurrencylinting.md- Concurrency-focused lint rules and SwiftLint async_without_awaitproduction-pitfalls.md- Silent data loss, cancellation gaps, migration bridges, .task vs onAppear
Project-Specific References (Anytype)
approachable-concurrency.md- Approachable concurrency quick guideswift-6-2-concurrency.md- Swift 6.2 concurrency updates (future reference)swiftui-concurrency-tour.md- SwiftUI-specific concurrency patterns
Best Practices Summary
- Prefer structured concurrency - Use task groups over unstructured tasks when possible
- Minimize suspension points - Keep actor-isolated sections small to reduce context switches
- Use @MainActor judiciously - Only for truly UI-related code
- Make types Sendable - Enable safe concurrent access by conforming to Sendable
- Handle cancellation - Check Task.isCancelled in long-running operations
- Avoid blocking - Never use semaphores or locks in async contexts (violates runtime contract)
- Test concurrent code - Use proper async test methods and consider timing issues
- Batch MainActor hops - Group UI updates to minimize context switches to/from main thread
- Understand the runtime contract - Threads must always make forward progress; use safe primitives
- Use LIBDISPATCH_COOPERATIVE_POOL_STRICT=1 - Debug environment variable to catch blocking calls
Verification Checklist (When You Change Concurrency Code)
- Confirm build settings (default isolation, strict concurrency, upcoming features) before interpreting diagnostics.
- After refactors:
- Run tests, especially concurrency-sensitive ones (see
references/testing.md). - If performance-related, verify with Instruments (see
references/performance.md). - If lifetime-related, verify deinit/cancellation behavior (see
references/memory-management.md).
- Run tests, especially concurrency-sensitive ones (see
Further Reading
- Fucking Approachable Swift Concurrency - Office Building mental model
- Swift Concurrency Course - Comprehensive reference (Antoine van der Lee)
- WWDC21: Swift Concurrency: Behind the Scenes - Runtime internals, threading model, runtime contract
- WWDC23: Beyond the basics of structured concurrency - Task groups, cancellation handlers, task locals
- Matt Massicotte's Blog - Deep dives on concurrency
- Swift Concurrency Documentation
Related Skills & Docs
- ios-dev-guidelines →
IOS_DEVELOPMENT_GUIDE.md- General Swift/iOS patterns, MVVM, Coordinators - tests-developer → Testing async code with Swift Testing framework
- swiftui-performance-developer → Performance optimization in SwiftUI views
Navigation: This skill provides concurrency mental models. For general Swift/iOS patterns, see ios-dev-guidelines.
Attribution: Office Building mental model from Dimillian/Skills. Reference files from AvdLee/Swift-Concurrency-Agent-Skill.
More from anyproto/anytype-swift
tests-developer
Smart router to testing patterns and practices. Use when writing unit tests, creating mocks, testing edge cases, or working with Swift Testing and XCTest frameworks.
2localization-developer
Context-aware routing to the Anytype iOS localization system. Use when working with .xcstrings files, Loc constants, hardcoded strings, or user-facing text.
2liquid-glass-developer
Context-aware routing to iOS 26 Liquid Glass implementation patterns. Use when working with glass effects, GlassEffectContainer, morphing transitions, or iOS 26 visual effects.
2skills-manager
Context-aware routing to skills and hooks management. Use when troubleshooting skill activation, fine-tuning keywords, or managing the automated documentation system.
1