16

在我的应用程序中,我试图通过将 Facebook 会话信息(令牌和到期日期)传递到我的服务器来进行身份验证(登录)。

我到应用服务器的登录顺序如下:

  1. 获取活跃的 facebook 会话。
  2. 如果会话有效,则获取令牌并将其发送到服务器。

在代码中:

Session session = Session.getActiveSession();
if(session != null && session.isOpened() && !didLogin) {
    didLogin = true;
    String token = session.getAccessToken();
    Date expires = session.getExpirationDate();
    loginWithFacebookSession(token, expires);
}

我突然注意到,经过几个月的正常工作后,发送到服务器的信息偶尔无效,特别token是空字符串和expires无效日期。

在浏览了 Facebook SDK(版本 3.0.1)之后,我发现了可能是我的错误的基础:

private static final Date MIN_DATE = new Date(Long.MIN_VALUE);
private static final Date ALREADY_EXPIRED_EXPIRATION_TIME = MIN_DATE;
private static final Date DEFAULT_LAST_REFRESH_TIME = new Date();

static AccessToken createEmptyToken(List<String> permissions) {
    return new AccessToken("", ALREADY_EXPIRED_EXPIRATION_TIME, permissions, AccessTokenSource.NONE,
            DEFAULT_LAST_REFRESH_TIME);
}

这意味着在此过程中,Facebook SDK 正在创建一个空令牌并使用SessionState.Category.OPENED_CATEGORY.

实际上没有信息时为什么要session.isOpened()返回?我应该检查不同的财产吗?这是 Facebook 的 SDK 中的错误吗?trueaccessToken

编辑:
向 Facebook 报告: https ://developers.facebook.com/bugs/121924628017965

4

3 回答 3

2

使用此方法并检查您的哈希是否正确

public void getHashKeyForFacebook(Activity activity, String packageName){
    try{
        PackageInfo info = activity.getPackageManager().getPackageInfo(packageName,  PackageManager.GET_SIGNATURES);

        for (Signature signature : info.signatures){
                MessageDigest md = MessageDigest.getInstance("SHA");
                md.update(signature.toByteArray());
                Log.d("KeyHash:", Base64.encodeToString(md.digest(), Base64.DEFAULT));
        }
    } catch (Exception ex){

    }
}
于 2013-09-07T05:53:29.500 回答
1

除了 Facebook SDK 像巫术一样,还有一些你正在做的事情我在我的身份验证流程中没有做。我在这里假设您的代码片段不在 Session.StatusCallback 调用函数中。无论如何,我总是在从回调中访问令牌之前打开会话进行读取。这个请求只是在返回之前使用加载微调器给用户带来几毫秒的不便。如果用户从他们的 Facebook 设置页面中删除了权限,它也会有所帮助。

我的身份验证流程是这样的:

private void startFbAuth() {
    Session session = Session.getActiveSession();
    if (session == null) {
        session = new Session(getApplicationContext());
        Session.setActiveSession(session);

        Session.OpenRequest openReadRequest = new Session.OpenRequest(this);
        openReadRequest.setPermissions(Arrays.asList({ "email" }));
        openReadRequest.setCallback(statusCallback);

        Session.NewPermissionsRequest permissionReadRequest = new Session.NewPermissionsRequest(this, Arrays.asList(EMAIL_PERMISSIONS));
        permissionReadRequest.setCallback(statusCallback);

        if (!session.isOpened() && !session.isClosed()) {
            session.openForRead(openReadRequest);
        } else {
            session.requestNewReadPermissions(permissionReadRequest);
        }
    }

    private Session.StatusCallback statusCallback = new SessionStatusCallback();

    private class SessionStatusCallback implements Session.StatusCallback {
        @Override
        public void call(Session session, SessionState state, Exception exception) {
            if (session.isClosed()) {
                session.closeAndClearTokenInformation();
                Session.setActiveSession(null);
                SessionStore.clearFacebookInformation(Activity.this);
                // Either throw an error or try reconnecting by calling startFbAuth
            }
            if (exception != null) {
                session.closeAndClearTokenInformation();
                Session.setActiveSession(session);
                // Either throw an error or try reconnecting by calling startFbAuth
            } else {
                if (session.isOpened()) {
                    String token = session.getAccessToken();
                    // It shouldn't be null or empty here
                }
            }
        }
    }
于 2013-06-26T05:48:54.197 回答
0

我解决了这个问题。问题出在关键哈希中。

按照本教程

转到第 5 步。4 运行样本。

最后,您将找到故障排除部分,这就是解决方案。

谢谢! https://developers.facebook.com/docs/getting-started/facebook-sdk-for-android/3.0/

于 2013-06-20T10:19:11.613 回答