I’ve finally resolved this issue after many hours of debugging — posting the full solution here in case it helps others.
✅ Problem
Since iOS 18.4, calling HomeKit APIs (like toggling accessories) from an AppIntent triggered by a Widget fails with the following error:
Error Domain=HMErrorDomain Code=80 "Missing entitlement for API."
UserInfo={
NSLocalizedFailureReason=Handler does not support background access,
NSLocalizedDescription=Missing entitlement for API.
}
This happens only when the intent runs inside the widget process, not when it runs in the main app (even in background).
✅ Solution Part 1: Force AppIntent to run in main app process
To bypass the limitations of the widget process, I made my AppIntent run in the main app instead, using the ForegroundContinuableIntent protocol.
You can do this by adding the following extension:
| @available(iOSApplicationExtension, unavailable) |
| extension MyActionIntent: ForegroundContinuableIntent {} |
Then declare your AppIntent as usual:
| struct MyActionIntent: AppIntent { |
| |
| init() {} |
| |
| static var openAppWhenRun: Bool { |
| return false |
| } |
| |
| func perform() async throws -> some IntentResult { |
| |
| return .result() |
| } |
| } |
With this, even when triggered from a widget, the intent runs in the app's process — which has proper entitlements and access to HomeKit.
🧨 BUT... another hidden problem was causing crashes
After applying the above fix, the intent still didn't seem to work — or worse, the app was crashing silently when launched in background.
After deeper investigation, I found the real cause:
I had a SwiftUI .backgroundTask(.appRefresh("MyBGTaskID")) { ... } modifier
declared on the WindowGroup.
Apparently, when the app is launched in background by an AppIntent, this SwiftUI modifier behaves incorrectly — the background task is not properly registered, and causes a crash when triggered.
But this crash is invisible if you test by launching the app manually — because in that case, everything works fine, and the .backgroundTask behaves normally.
✅ Solution Part 2: Register background task manually
I fixed this issue by removing the SwiftUI .backgroundTask modifier and manually registering the background task the "classic" way, in the init() of the @main app struct:
| import BackgroundTasks |
| |
| @main |
| struct MyApp: App { |
| |
| init() { |
| BGTaskScheduler.shared.register(forTaskWithIdentifier: "MyBGTaskID", using: nil) { task in |
| |
| } |
| } |
| |
| var body: some Scene { |
| WindowGroup { |
| ContentView() |
| } |
| } |
| } |
✅ Summary
iOS 18.4 introduced stricter sandboxing for AppIntents running inside Widgets.
Use ForegroundContinuableIntent to force your AppIntent to run in the main app context (where HomeKit works).
Beware of SwiftUI .backgroundTask modifiers — they may cause background crashes when the app is launched by an AppIntent.
Use manual registration via BGTaskScheduler.shared.register(...) instead.
Let me know if anyone else experienced the same issue, or if you found a better way to debug background crashes in SwiftUI contexts.
Hope this helps! 🙌