Unable to read app group's NSUserDefaults but able to write on it

672 Views Asked by At

I have a really strange problem while developing a content blocker extension. I'm trying to share data between it and my app using NSUserDefaults with an app group but but reading something from the app group always crashes the extension.

Here is a sample code that is working:

func beginRequestWithExtensionContext(context: NSExtensionContext) {
    NSUserDefaults(suiteName: "group.Bug")!.setBool(true, forKey: "here")
    NSUserDefaults(suiteName: "group.Bug")!.synchronize()

    let attachment = NSItemProvider(contentsOfURL: NSBundle.mainBundle().URLForResource("blockerList", withExtension: "json"))!
    let item = NSExtensionItem()
    item.attachments = [attachment]
    context.completeRequestReturningItems([item], completionHandler: nil);
}

The extension doesn't return an error and I can then read here in my app.

Here is the same thing with one more line trying to read the value I just settled:

func beginRequestWithExtensionContext(context: NSExtensionContext) {
    NSUserDefaults(suiteName: "group.Bug")!.setBool(true, forKey: "here")
    NSUserDefaults(suiteName: "group.Bug")!.synchronize()
    NSLog("%@", (NSUserDefaults(suiteName: "group.Bug")!.boolForKey("here")))

    let attachment = NSItemProvider(contentsOfURL: NSBundle.mainBundle().URLForResource("blockerList", withExtension: "json"))!
    let item = NSExtensionItem()
    item.attachments = [attachment]
    context.completeRequestReturningItems([item], completionHandler: nil);
}

It returns Optional(Error Domain=ContentBlockerErrorDomain Code=3 "(null)").

The problem only happens when running the code on a real device, it's working on the simulator. Does someone know what could be the source of the problem?

2

There are 2 best solutions below

2
On

When writing data to NSUserDefaults, don't forget NSUserDefaults( ).synchronize()

0
On

Don't read the value back immediately after writing to user defaults because first of all the value is known in code and second of all the process to synchronize the database works asynchronously.

That means reading the value back at once is unreliable.