Аутентификация X-FACEBOOK-PLATFORM с библиотекой SMACK Java с использованием OAuth 2.0

Первый пост здесь, поэтому, пожалуйста, будьте нежны. Я создаю клиент чата facebook, используя библиотеку Smack. Я использую метод X-FACEBOOK-PLATFORM, чтобы не сохранять пароли. У меня он работал правильно, используя oauth 1.0, и я хочу изменить его на 2.0 из-за крайнего срока 1 октября; p. Насколько я понимаю, единственное, что мне нужно сделать для перехода на 2.0, — это удалить параметры «sig» и «session_key» и добавить параметр «access_token», но я получаю «Аутентификация SASL X-FACEBOOK- ПЛАТФОРМА не удалась: авторизация не разрешена:". Я использую этот пользовательский класс SASLMechanism:

package smackresearching;
import java.io.IOException;
import java.net.URLEncoder;
import java.util.GregorianCalendar;
import java.util.HashMap;
import java.util.Map;

import javax.security.sasl.Sasl;

import org.jivesoftware.smack.SASLAuthentication;
import org.jivesoftware.smack.XMPPException;
import org.jivesoftware.smack.sasl.SASLMechanism;
import org.jivesoftware.smack.util.Base64;

public class SASLXFacebookPlatformMechanism extends SASLMechanism {

    public static final String NAME = "X-FACEBOOK-PLATFORM";
    private String apiKey = "";
        private String accessToken = "";


    /**
     * Constructor.
     */
    public SASLXFacebookPlatformMechanism(SASLAuthentication saslAuthentication) {
            super(saslAuthentication);
    }

    @Override
    protected void authenticate() throws IOException, XMPPException {
        // Send the authentication to the server
        getSASLAuthentication().send(new AuthMechanism(getName(), ""));
    }

    @Override
    public void authenticate(String apiKey, String host, String accessToken) throws IOException, XMPPException {
        this.apiKey = apiKey;
        this.accessToken = accessToken;
        this.hostname = host;

        String[] mechanisms = { "DIGEST-MD5" };
        Map<String, String> props = new HashMap<String, String>();
        this.sc = Sasl.createSaslClient(mechanisms, null, "xmpp", host, props, this);
        authenticate();
    }

    @Override
    protected String getName() {
            return NAME;
    }

    @Override
    public void challengeReceived(String challenge) throws IOException {
        byte[] response = null;

        if (challenge != null) {
                    String decodedChallenge = new String(Base64.decode(challenge));
                    Map<String, String> parameters = getQueryMap(decodedChallenge);

                    String version = "1.0";
                    String nonce = parameters.get("nonce");
                    String method = parameters.get("method");

                    long callId = new GregorianCalendar().getTimeInMillis() / 1000L;

                    String composedResponse = "api_key=" + URLEncoder.encode(apiKey, "utf-8")
                                                                            + "&call_id=" + callId
                                                                            + "&method=" + URLEncoder.encode(method, "utf-8")
                                                                            + "&nonce=" + URLEncoder.encode(nonce, "utf-8")
                                                                            + "&access_token=" + URLEncoder.encode(accessToken, "utf-8")
                                                                            + "&v=" + URLEncoder.encode(version, "utf-8");

                    response = composedResponse.getBytes("utf-8");
        }

        String authenticationText = "";

        if (response != null){
                    authenticationText = Base64.encodeBytes(response, Base64.DONT_BREAK_LINES);
                }
        // Send the authentication to the server
        getSASLAuthentication().send(new Response(authenticationText));
    }

    private Map<String, String> getQueryMap(String query) {
            Map<String, String> map = new HashMap<String, String>();
            String[] params = query.split("\\&");

            for (String param : params) {
                    String[] fields = param.split("=", 2);
                    map.put(fields[0], (fields.length > 1 ? fields[1] : null));
            }
            return map;
    }
    }

А это основной код:

ConnectionConfiguration config = new ConnectionConfiguration("chat.facebook.com",     5222);
config.setSASLAuthenticationEnabled(true);
//        config.setDebuggerEnabled(true);
XMPPConnection connection = new XMPPConnection(config);


try {
    //ESTA LINEA HACE QUE NO DE TIMEOUT
    SmackConfiguration.setPacketReplyTimeout(15000);
    XMPPConnection.DEBUG_ENABLED = true;
    SASLAuthentication.registerSASLMechanism("X-FACEBOOK-PLATFORM", SASLXFacebookPlatformMechanism.class);
    SASLAuthentication.supportSASLMechanism("X-FACEBOOK-PLATFORM", 0);
    connection.connect();
    String apiKey = "3290282390339";
    String accessToken = "ADSJGFGFKDKSJKSJD0-43DKJSDJKSDKJSD094JJSDKJSDKJDSNDSKJEWEWNSDLkljdkjs";
    connection.login(apiKey, accessToken);
...

Заранее благодарю за любой совет.


person alscu    schedule 12.09.2011    source источник


Ответы (1)


Отсутствует амперсанд для параметра access_token в строке составленного ответа. Это опечатка?

Не могли бы вы опубликовать аутентификационный текст, который вы отправляете?

person Alexcode    schedule 13.09.2011
comment
Да, Александр прав, поместите &access_token, и он будет работать как шарм ;-) - person Guillaume; 13.09.2011
comment
Мне ОЧЕНЬ жаль, что я побеспокоил всех вас опечаткой. Это действительно была ошибка, работает отлично. Надеюсь, этот код может быть полезен другим людям. - person alscu; 13.09.2011
comment
@alscu, не волнуйтесь, эти ошибки хуже, вы можете потратить часы без удачи, а свежий взгляд найдет это за секунды. Рад, что это работает. - person Alexcode; 13.09.2011
comment
@alscu, и я хотел бы добавить спасибо, потому что я пытался подключиться таким образом к Facebook в течение некоторого времени, но у меня не было того же кода, что и у вас, и ваш код помог! так что спасибо за ваш пост! - person Guillaume; 18.09.2011