我有一个可以拥有多个用户的iOS应用程序,我想将Facebook集成到其中。我想缓存每个用户的登录凭据。用户将拥有应用程序登录,因此每个用户只能使用自己的Facebook信用卡。 Facebook的文档声明您可以通过覆盖FBSessionTokenCachingStrategy对象来管理多个用户的会话令牌:
https://developers.facebook.com/docs/ios/login-tutorial/
我正在使用本地缓存方法,我让它在用户登录我的应用程序的地方工作,然后如果他们有相关的Facebook令牌,我将它们登录到那里会话。否则,我提示他们使用Facebook的API输入Facebook凭据,将用户重定向到Facebook网页登录页面。
[FBSession openActiveSessionWithReadPermissions:@[@"basic_info"]
allowLoginUI:YES
completionHandler:
^(FBSession *session, FBSessionState state, NSError *error)
{
[self loginToFacebookEnd:session state:state error:error];
}];
上述方法适用于第一用户。登录的第二个用户按预期重定向到Facebook Web登录页面,但第一个用户现在通过safari登录,第二个用户现在使用第一个用户凭据登录。
我希望能够让用户不必登录safari,或者能够使用类似嵌入式Web对话框的内容。然而,根据Facebook API文档,嵌入式对话框似乎不允许我在应用重启后加载用户令牌:
https://developers.facebook.com/docs/ios/login-tutorial/#control
“嵌入式WebView登录对话框的缺点:人们必须填写 他们每次通过登录流程时都会登录凭据。“
此外,当从嵌入式WebView保存令牌时,调用以下方法会清除我的令牌缓存:
FBSession *session = [[FBSession alloc] initWithAppID:nil
permissions:@[@"basic_info"]
urlSchemeSuffix:nil
tokenCacheStrategy:tokenCaching];
有没有人知道如何支持多个用户从iOS应用登录Facebook而没有第一个用户的凭据在Safari中缓存?
更新1
好的,我知道现在可以使用Embedded WebView处理多个用户,因为Facebook包含的示例程序之一(SwitchUserSample)就是这样做的。加载我的FBSession似乎有些问题,因为很多值都是null,尽管我调用的代码与示例完全相同。有谁知道会导致会话恢复的原因如下:
<FBSession: 0x16d9f3a0, state: FBSessionStateCreated, loginHandler: 0x0, appID: 658013034244859, urlSchemeSuffix: , tokenCachingStrategy:<FBSessionTokenCachingStrategy: 0x16d56190>, expirationDate: (null), refreshDate: (null), attemptedRefreshDate: 0001-12-30 00:00:00 +0000, permissions:(null)>
如果我使用除FBSessionLoginBehaviorForcingWebView以外的任何行为,那么我将其作为我的反序列化令牌:
<FBSession: 0x155823d0, state: FBSessionStateOpen, loginHandler: 0x73364, appID: 658013034244859, urlSchemeSuffix: , tokenCachingStrategy:<FBSessionTokenCachingStrategy: 0x15577c90>, expirationDate: 2014-02-15 20:01:50 +0000, refreshDate: 2013-12-17 20:01:51 +0000, attemptedRefreshDate: 0001-12-30 00:00:00 +0000, permissions:(
"user_birthday",
"user_friends"
)>
更新2
我发现这肯定与我的令牌缓存方式有关。我已经复制了Facebook示例正在使用的方法,但我仍然在FBSession中获取我的日期的空值。以下是我调用的缓存方法:
+ (FBSessionTokenCachingStrategy*)createCachingStrategyForSlot:(int)slot {
FBSessionTokenCachingStrategy *tokenCachingStrategy = [[FBSessionTokenCachingStrategy alloc]
initWithUserDefaultTokenInformationKeyName:[NSString stringWithFormat:@"SUUserTokenInfo%d", slot]];
return tokenCachingStrategy;
}
+ (FBSession*)createSessionForSlot:(int)slot {
FBSessionTokenCachingStrategy *tokenCachingStrategy = [self createCachingStrategyForSlot:slot];
FBSession *session = [[FBSession alloc] initWithAppID:nil
permissions:@[@"basic_info",@"user_birthday"]
urlSchemeSuffix:nil
tokenCacheStrategy:tokenCachingStrategy];
return session;
}
以下是我的称呼方式:
+ (BOOL)openSessionWithAllowLoginUI:(BOOL)allowLoginUI
{
BOOL openSessionResult = NO;
FBSession *session = [self createSessionForSlot:1];
// If showing the login UI, or if a cached token is available, then open the session.
if (allowLoginUI || session.state == FBSessionStateCreatedTokenLoaded)
{
// For debugging purposes log if cached token was found
if (session.state == FBSessionStateCreatedTokenLoaded)
{
NSLog(@"Cached token found.");
}
// Set the active session
[FBSession setActiveSession:session];
// If the session state is not any of the two "open" states when the button is clicked
if (FBSession.activeSession.state != FBSessionStateOpen && FBSession.activeSession.state != FBSessionStateOpenTokenExtended)
{
// Open the session.
[session openWithBehavior:FBSessionLoginBehaviorForcingWebView
completionHandler:^(FBSession *session,
FBSessionState state,
NSError *error)
{
[SocialInteraction loginToFacebookEnd:session state:state error:error];
}];
}
// Return the result - will be set to open immediately from the session
// open call if a cached token was previously found.
openSessionResult = session.isOpen;
}
return openSessionResult;
}
令人沮丧的是,如果我将登录行为从FBSessionLoginBehaviorForcingWebView更改为FBSessionLoginBehaviorWithFallbackToWebView,则令牌正确加载!但是,这种行为对我的应用程序不起作用。必须有一些我想要的其他设置,让嵌入式WebView缓存其令牌。
答案 0 :(得分:0)
我正在将其从评论转移到答案,以便我可以输入更多内容:
如果您的FBSession
个实例的状态为FBSessionStateCreated
,那么您将不会拥有expirationDate
个属性等。根据我的经验,它必须是有效开放的(FBSessionStateCreatedTokenLoaded
,FBSessionStateOpen
,FBSessionStateOpenTokenExtended
)。您的FBSession对象是否具有与之关联的accessToken?如果是,则继续尝试打开会话,查看状态是否发生变化,并获得曾经为NULL的值。
当我有效FBSession
我通常使用时,打开accessToken
:
[FBSession openActiveSessionWithAllowLoginUI:NO];
更新1:
好的,这是Web视图的一种变通方法 - 如果您使用Safari,您可以使用下面的代码清除用户的cookie。我可能会在cookie的域中搜索Facebook.com
或fb.com
(可能需要对FB cookie域进行一些测试):
NSHTTPCookie *cookie;
NSHTTPCookieStorage *storage = [NSHTTPCookieStorage sharedHTTPCookieStorage];
for (cookie in [storage cookies]) {
[storage deleteCookie:cookie];
}
[[NSUserDefaults standardUserDefaults] synchronize];