Why does iOS get a new identifierForVendor when app updates?

10,564

There was a bug affecting the calculation of identifierForVendor when updating an app from app store between May and July. Apple has claimed they have already solved the issue and pushing another update should restore the original value before the critical date. Reference: https://openradar.appspot.com/22677034

Take note that some users still have observed this issue even updating weeks after July. So it is still possible that bug is still there for some or it could resurface anytime in the future. So if you are using this data as part of your encryption, best is to save this to the keychain.

Share:
10,564
Esqarrouth
Author by

Esqarrouth

World-class code copy paster.

Updated on August 05, 2022

Comments

  • Esqarrouth
    Esqarrouth over 1 year

    Every time my app is updated from the App Store some small number of the users get a new identifierForVendor for some reason. My users don't sign up or login. They are all anonymous so I need to separate them through their vendor IDs.

    I've considered that there could've been insufficient space on some devices, resulting in the app being deleted and reinstalled, but that's not the case since in the last update a friend of mine had over 2GB of empty space.

    I know that the identifierForVendor is changed for a user who deletes and reinstalls the app. But that's not the case here, as the app is just updated.

    What could the problem be? The weird part is that this hasn't happened for development team devices yet. Or there are still users who haven't experienced this bug after countless app updates, OS updates, etc. This only happens to a small percentage of the users. All users are iOS7+ and it happens for different device models and iOS versions.

    I use this code to get their ID:

    static let DeviceId = UIDevice.currentDevice().identifierForVendor.UUIDString
    

    Then I save them into NSUserDefaults:

    NSUserDefaults.standardUserDefaults().setBool(true, forKey: "User" + DeviceId)
    NSUserDefaults.standardUserDefaults().synchronize()
    

    Then I check if the user exists, at every new login:

    static func doesUserExist() -> Bool {
        var userDefaultValue: AnyObject? = NSUserDefaults.standardUserDefaults().valueForKey("User" + DeviceId)
    
        if defaultValue == true {
            println("Userdefaults already has this guy, moving on")
            FirstTime = false
            return true
        } else {
            println("First time in the app!")
            FirstTime = true
            return false
        }
    }
    

    If the user does exist it starts the login process. If the user does not exist it shows them the signup process. I am using Parse.com as backend and the deviceID is used as a username. When that small amount of users experience this bug, I see a new username and a new account created.

  • Esqarrouth
    Esqarrouth almost 9 years
    thank you for the code. it will help solve this problem. however this post is focused more on what the problem might be, instead of a solution.
  • Jonny
    Jonny almost 9 years
    As mentioned in the comments to this answer: stackoverflow.com/a/21878670/129202 synchronization of keychain using iCloud breaks this.
  • dogsgod
    dogsgod almost 9 years
    @Jonny: you could solve that by adding some device information to the key. But for the question here, that is not necessary, as it's only about identifying a user - I doubt there are many users sharing their iCloud ...
  • chalcopyrite
    chalcopyrite over 8 years
    If you are using the native security framework when saving to keychain, there is attribute to make a keychain item accessible only to the device. That is, it will not be synced to iTunes or iCloud. You can add to your attributes this value: CFDictionaryAddValue(attributes, kSecAttrAccessible, kSecAttrAccessibleWhenUnlockedThisDeviceOnly)
  • FranticRock
    FranticRock over 6 years
    I am testing in iOS11 and the same identifierForVendor is being returned after I explicitly un-install and re-install the app. I need to wait for a day to see if this persists over longer period of time. (Tested 3 un-installs and re-installs: same ID each time). Occurs both with Side-Load with Xcode managed profile, and ad-hoc builds signed with developer profile. Haven't tested yet with release-signed builds. We actually need a unique identifier for our purposes, so we will resort to using the keychain and UUID instead.