containerURLForSecurityApplicationGroupIdentifier

2019-01-26 03:44发布

问题:

Context

With iOS8 and extensions Apple introduced the App group containers (more info here).

Problem

We use it through NSFileManager's containerURLForSecurityApplicationGroupIdentifier: method. It works well in production on AppStore (with both iOS8 and iOS7). Problem is since our team updated to Xcode 6.1 (6A1052d), the method returns nil on Simulators.

We searched a lot but we didn't manage to find any clues. Even this question or this one are not applicable here.

TLDR: Question

Does NSFileManager's containerURLForSecurityApplicationGroupIdentifier: method works on your simulators? Is there any way to fix that?

回答1:

I was able to solve this problem on my side. The documentation says

Your app must have a com.apple.security.application-groups entitlement for the specified application group. https://developer.apple.com/library/mac/documentation/Cocoa/Reference/Foundation/Classes/NSFileManager_Class/index.html#//apple_ref/occ/instm/NSFileManager/containerURLForSecurityApplicationGroupIdentifier:

At some point in my app, I had removed the entitlements file from my app target/build settings/code signing/code signing entitlements . Both debug and release must include YOUR_APP.entitlements file . My debug wasn't including it so it wasn't working on simulator



回答2:

The method containerURLForSecurityApplicationGroupIdentifier: works in both device and Simulator.

If you have setup both targets (or Target and WatchKit Extension for instance) that are going to share data to support Groups, and it keeps returning nil, you may want to check your Targets settings.

  1. Target Settings > General > Team > yourCompany signing team should be selected here. If there is a warning and a Fix This Issue button, be good and fix the issue by hitting it :)

  2. Target Settings > Capabilities > App Groups > Is On + shows your group.com.yourcompany.yourapp.sharedContainter group name ("sharedContainer" is my choice name) is selected and is on black (not red). If it's red or there is a warning and Fix This Issue button, renew the group or create a different one.

  3. Target Settings > Build Settings > Code Signing > Code Signing Entitlements: points to your entitlements files for both Debug and Release (something like yourApp/yourApp.entitlements)

  4. Target Settings > Build Settings > Code Signing > Code Signing Identity: Make sure the identity is part of the Team selected above. I use iOS Developer.

  5. Target Settings > Build Settings > Code Signing > Provisioning Profile: As above, make sure the PP you are using is of the Team required. I set it to Automatic.

Check those for both targets sharing data.

I hope it helps



回答3:

It doesn't make it clear the the docs but I found that the group identifier is case sensitive.



回答4:

I find containerURLForSecurityApplicationGroupIdentifier can do work both on real device and simulator with xcode 6.2. I came across into this thread because i also got nil, but now i find the root cause is my typing error of the 'App Groups' name.

More, here is the list you need check:

  1. make sure 'app groups' been created either by xcode or web console at https://developer.apple.com
  2. make sure the 'app groups' is 'enabled' on 'Development' and 'Production' at 'web console: Identifiers->App IDs'
  3. make sure the matched 'Provisioning Profiles' are 'active'
  4. the groups name of entitlement in both containing app and extension app should match the one you created in step 1
  5. make sure the bundle identifier of all targets is the same as the one from 'web console: Identifiers->App IDs
  6. make sure the 'Team' inside of all targets matches the one from web console
  7. make sure to pick the correct group under 'Capabilities' tab on both containing app and extension app target, and no red color
  8. clean up xcode cache by cmd+shift+k
  9. rebuild and pray


回答5:

The Method containerURLForSecurityApplicationGroupIdentifier: works in my Simulator. But if i delete all files in the group folder (in case a user performs a logout in my app) the method returns nil on the next RUN in Xcode 6.1. I tested this with Xcode 6.2 and iOS SDK 8.2 Beta as well but it doesn't work.

The Code runs fine on a real device.

I also tried the above solutions without success.



回答6:

For me there was some mess with my iOS Team Provisioning Profile (which is not required to run in Simulator).

In order to run my app on my iPhone I had to first reconfigure the iOS Team Provisioning Profile for the Debug environment. Calling containerURLForSecurityApplicationGroupIdentifier: worked perfectly fine on the device. Afterwards I went back to the Simulator and - there you go - it worked again as well.



回答7:

I've found that the reason of this error in my case was a file ".com.apple.mobile_container_manager.metadata.plist". After I've got my file from this directory, I deleted all files in it. So this file was deleted also. And when you delete this file, app returns you a nil for containerURLForSecurityApplicationGroupIdentifier

So I changed my code to this:

    NSURL *groupPath = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:YGGroupIdentifier];
    NSArray *fileArray = [fileManager contentsOfDirectoryAtPath:[groupPath path] error:nil];
    for (NSString *filename in fileArray)  {
        if (![filename hasPrefix:@".com"]) {
            [fileManager removeItemAtPath:[[groupPath path] stringByAppendingPathComponent:filename] error:nil];
        }
    }

Now it works great, even in simulator.