iOS開發(fā)-獲取設備號,鑰匙串keychain 記錄獲取

在之前開發(fā)游戲官方渠道sdk的時候,通常會有快速登錄的需求,這時候就需要獲取設備的唯一標識 ?來當做用戶的賬號。

在一些大公司,比如網易、騰訊、游久等,他們都會有自己的一套獲取設備號的方法。


之前試過一些,比如mac地址,uuid ,udid等,這些方式也都被蘋果禁用了。

當時,uuid可以獲取到,但是如果用戶在設置中 ??

設置-隱私-廣告-跟蹤廣告,這個如果是打開狀態(tài)的話,會造成獲取不到設備號(每次設備號都會重新生成)。所以,游戲玩家進入游戲,下線再次登錄之后,會發(fā)現(xiàn)之前的游戲角色不見了。(或者變成別人的賬號,發(fā)生串號問題)


那么,我們就要獲取一個永不會變的“標識符”,存在設備的鑰匙串中,即使刪除應用程序(游戲),這個“標識符”依然存在。


如下:項目中實際的解決方案。

+ (NSString*)UDID

{

NSString *udid = [SvUDIDTools getUDIDFromKeyChain];

if(!udid) {

NSString *sysVersion = [UIDevice currentDevice].systemVersion;

CGFloat version = [sysVersion floatValue];

if(version >=7.0) {

udid = [SvUDIDTools _UDID_iOS7];

}

elseif(version >=2.0) {

udid = [SvUDIDTools _UDID_iOS6];

}

[SvUDIDTools settUDIDToKeyChain:udid];

}

returnudid;

}

/*

* iOS 6.0

* use wifi's mac address

*/

+ (NSString*)_UDID_iOS6

{

return[SvUDIDTools getMacAddress];

}

/*

* iOS 7.0

* Starting from iOS 7, the system always returns the value 02:00:00:00:00:00

* when you ask for the MAC address on any device.

* use identifierForVendor + keyChain

* make sure UDID consistency atfer app delete and reinstall

*/

+ (NSString*)_UDID_iOS7

{

return[[UIDevice currentDevice].identifierForVendor UUIDString];

}

#pragma mark -

#pragma mark Helper Method for Get Mac Address

// fromhttp://stackoverflow.com/questions/677530/how-can-i-programmatically-get-the-mac-address-of-an-iphone

+ (NSString *)getMacAddress

{

intmgmtInfoBase[6];

char*msgBuffer =NULL;

size_tlength;

unsignedcharmacAddress[6];

structif_msghdr*interfaceMsgStruct;

structsockaddr_dl*socketStruct;

NSString*errorFlag =nil;

// Setup the management Information Base (mib)

mgmtInfoBase[0] = CTL_NET;// Request network subsystem

mgmtInfoBase[1] = AF_ROUTE;// Routing table info

mgmtInfoBase[2] =0;

mgmtInfoBase[3] = AF_LINK;// Request link layer information

mgmtInfoBase[4] = NET_RT_IFLIST;// Request all configured interfaces

// With all configured interfaces requested, get handle index

if((mgmtInfoBase[5] = if_nametoindex("en0")) ==0)

errorFlag =@"if_nametoindex failure";

else

{

// Get the size of the data available (store in len)

if(sysctl(mgmtInfoBase,6,NULL, &length,NULL,0) <0)

errorFlag =@"sysctl mgmtInfoBase failure";

else

{

// Alloc memory based on above call

if((msgBuffer = malloc(length)) ==NULL)

errorFlag =@"buffer allocation failure";

else

{

// Get system information, store in buffer

if(sysctl(mgmtInfoBase,6, msgBuffer, &length,NULL,0) <0)

errorFlag =@"sysctl msgBuffer failure";

}

}

}

// Befor going any further...

if(errorFlag !=NULL)

{

NSLog(@"Error: %@", errorFlag);

if(msgBuffer) {

free(msgBuffer);

}

returnerrorFlag;

}

// Map msgbuffer to interface message structure

interfaceMsgStruct = (structif_msghdr *) msgBuffer;

// Map to link-level socket structure

socketStruct = (structsockaddr_dl *) (interfaceMsgStruct +1);

// Copy link layer address data in socket structure to an array

memcpy(&macAddress, socketStruct->sdl_data + socketStruct->sdl_nlen,6);

// Read from char array into a string object, into traditional Mac address format

NSString *macAddressString = [NSString stringWithFormat:@"%02X:%02X:%02X:%02X:%02X:%02X",

macAddress[0], macAddress[1], macAddress[2],

macAddress[3], macAddress[4], macAddress[5]];

NSLog(@"Mac Address: %@", macAddressString);

// Release the buffer memory

free(msgBuffer);

returnmacAddressString;

}

#pragma mark -

#pragma mark Helper Method for make identityForVendor consistency

+ (NSString*)getUDIDFromKeyChain

{

NSMutableDictionary *dictForQuery = [[NSMutableDictionary alloc] init];

[dictForQuery setValue:(id)kSecClassGenericPassword forKey:(id)kSecClass];

// set Attr Description for query

[dictForQuery setValue:[NSString stringWithUTF8String:kKeychainUDIDItemIdentifier]

forKey:kSecAttrDescription];

// set Attr Identity for query

NSData *keychainItemID = [NSData dataWithBytes:kKeychainUDIDItemIdentifier

length:strlen(kKeychainUDIDItemIdentifier)];

[dictForQuery setObject:keychainItemID forKey:(id)kSecAttrGeneric];

// The keychain access group attribute determines if this item can be shared

// amongst multiple apps whose code signing entitlements contain the same keychain access group.

NSString *accessGroup = [NSString stringWithUTF8String:kKeyChainUDIDAccessGroup];

if(accessGroup !=nil)

{

#if TARGET_IPHONE_SIMULATOR

// Ignore the access group if running on the iPhone simulator.

//

// Apps that are built for the simulator aren't signed, so there's no keychain access group

// for the simulator to check. This means that all apps can see all keychain items when run

// on the simulator.

//

// If a SecItem contains an access group attribute, SecItemAdd and SecItemUpdate on the

// simulator will return -25243 (errSecNoAccessForItem).

#else

[dictForQuery setObject:accessGroup forKey:(id)kSecAttrAccessGroup];

#endif

}

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecMatchCaseInsensitive];

[dictForQuery setValue:(id)kSecMatchLimitOne forKey:(id)kSecMatchLimit];

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecReturnData];

OSStatus queryErr= noErr;

NSData*udidValue =nil;

NSString *udid=nil;

queryErr = SecItemCopyMatching((CFDictionaryRef)dictForQuery, (CFTypeRef*)&udidValue);

NSMutableDictionary *dict =nil;

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecReturnAttributes];

queryErr = SecItemCopyMatching((CFDictionaryRef)dictForQuery, (CFTypeRef*)&dict);

if(queryErr == errSecItemNotFound) {

NSLog(@"KeyChain Item: %@ not found!!!", [NSString stringWithUTF8String:kKeychainUDIDItemIdentifier]);

}

elseif(queryErr != errSecSuccess) {

NSLog(@"KeyChain Item query Error!!! Error code:%ld", queryErr);

}

if(queryErr == errSecSuccess) {

NSLog(@"KeyChain Item: %@", udidValue);

if(udidValue) {

udid = [NSString stringWithUTF8String:udidValue.bytes];

[udidValue release];

}

[dict release];

}

[dictForQuery release];

returnudid;

}

+ (BOOL)settUDIDToKeyChain:(NSString*)udid

{

NSMutableDictionary *dictForAdd = [[NSMutableDictionary alloc] init];

[dictForAdd setValue:(id)kSecClassGenericPassword forKey:(id)kSecClass];

[dictForAdd setValue:[NSString stringWithUTF8String:kKeychainUDIDItemIdentifier] forKey:kSecAttrDescription];

[dictForAdd setValue:@"UUID"forKey:(id)kSecAttrGeneric];

// Default attributes for keychain item.

[dictForAdd setObject:@""forKey:(id)kSecAttrAccount];

[dictForAdd setObject:@""forKey:(id)kSecAttrLabel];

// The keychain access group attribute determines if this item can be shared

// amongst multiple apps whose code signing entitlements contain the same keychain access group.

NSString *accessGroup = [NSString stringWithUTF8String:kKeyChainUDIDAccessGroup];

if(accessGroup !=nil)

{

#if TARGET_IPHONE_SIMULATOR

// Ignore the access group if running on the iPhone simulator.

//

// Apps that are built for the simulator aren't signed, so there's no keychain access group

// for the simulator to check. This means that all apps can see all keychain items when run

// on the simulator.

//

// If a SecItem contains an access group attribute, SecItemAdd and SecItemUpdate on the

// simulator will return -25243 (errSecNoAccessForItem).

#else

[dictForAdd setObject:accessGroup forKey:(id)kSecAttrAccessGroup];

#endif

}

constchar*udidStr = [udid UTF8String];

NSData *keyChainItemValue = [NSData dataWithBytes:udidStr length:strlen(udidStr)];

[dictForAdd setValue:keyChainItemValue forKey:(id)kSecValueData];

OSStatus writeErr = noErr;

if([SvUDIDTools getUDIDFromKeyChain]) {// there is item in keychain

[SvUDIDTools updateUDIDInKeyChain:udid];

[dictForAdd release];

returnYES;

}

else{// add item to keychain

writeErr = SecItemAdd((CFDictionaryRef)dictForAdd,NULL);

if(writeErr != errSecSuccess) {

NSLog(@"Add KeyChain Item Error!!! Error Code:%ld", writeErr);

[dictForAdd release];

returnNO;

}

else{

NSLog(@"Add KeyChain Item Success!!!");

[dictForAdd release];

returnYES;

}

}

[dictForAdd release];

returnNO;

}

+ (BOOL)removeUDIDFromKeyChain

{

NSMutableDictionary *dictToDelete = [[NSMutableDictionary alloc] init];

[dictToDelete setValue:(id)kSecClassGenericPassword forKey:(id)kSecClass];

NSData *keyChainItemID = [NSData dataWithBytes:kKeychainUDIDItemIdentifier length:strlen(kKeychainUDIDItemIdentifier)];

[dictToDelete setValue:keyChainItemID forKey:(id)kSecAttrGeneric];

OSStatus deleteErr = noErr;

deleteErr = SecItemDelete((CFDictionaryRef)dictToDelete);

if(deleteErr != errSecSuccess) {

NSLog(@"delete UUID from KeyChain Error!!! Error code:%ld", deleteErr);

[dictToDelete release];

returnNO;

}

else{

NSLog(@"delete success!!!");

}

[dictToDelete release];

returnYES;

}

+ (BOOL)updateUDIDInKeyChain:(NSString*)newUDID

{

NSMutableDictionary *dictForQuery = [[NSMutableDictionary alloc] init];

[dictForQuery setValue:(id)kSecClassGenericPassword forKey:(id)kSecClass];

NSData *keychainItemID = [NSData dataWithBytes:kKeychainUDIDItemIdentifier

length:strlen(kKeychainUDIDItemIdentifier)];

[dictForQuery setValue:keychainItemID forKey:(id)kSecAttrGeneric];

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecMatchCaseInsensitive];

[dictForQuery setValue:(id)kSecMatchLimitOne forKey:(id)kSecMatchLimit];

[dictForQuery setValue:(id)kCFBooleanTrue forKey:(id)kSecReturnAttributes];

NSDictionary *queryResult =nil;

SecItemCopyMatching((CFDictionaryRef)dictForQuery, (CFTypeRef*)&queryResult);

if(queryResult) {

NSMutableDictionary *dictForUpdate = [[NSMutableDictionary alloc] init];

[dictForUpdate setValue:[NSString stringWithUTF8String:kKeychainUDIDItemIdentifier] forKey:kSecAttrDescription];

[dictForUpdate setValue:keychainItemID forKey:(id)kSecAttrGeneric];

constchar*udidStr = [newUDID UTF8String];

NSData *keyChainItemValue = [NSData dataWithBytes:udidStr length:strlen(udidStr)];

[dictForUpdate setValue:keyChainItemValue forKey:(id)kSecValueData];

OSStatus updateErr = noErr;

// First we need the attributes from the Keychain.

NSMutableDictionary *updateItem = [NSMutableDictionary dictionaryWithDictionary:queryResult];

[queryResult release];

// Second we need to add the appropriate search key/values.

// set kSecClass is Very important

[updateItem setObject:(id)kSecClassGenericPassword forKey:(id)kSecClass];

updateErr = SecItemUpdate((CFDictionaryRef)updateItem, (CFDictionaryRef)dictForUpdate);

if(updateErr != errSecSuccess) {

NSLog(@"Update KeyChain Item Error!!! Error Code:%ld", updateErr);

[dictForQuery release];

[dictForUpdate release];

returnNO;

}

else{

NSLog(@"Update KeyChain Item Success!!!");

[dictForQuery release];

[dictForUpdate release];

returnYES;

}

}

[dictForQuery release];

returnNO;

}

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容