NTLM-аутентификация в Android - PullRequest
9 голосов
/ 07 января 2012

Я занимаюсь разработкой приложения для Android, которое обращается к веб-службе Sharepoint и выполняет обработку SOAP. Я пробовал разные способы, такие как JCIFS и т. Д. Ничего не помогало.

Может ли кто-нибудь помочь мне здесь? Я гуглю это много дней, но все, кто имеет эту проблему, разочарованы.

Спасибо, Индраджит

Ответы [ 2 ]

8 голосов
/ 08 февраля 2017

Я не специалист по NTLM, но я успешно подключился к нашему бэкэнду, используя библиотеку JCIFS и некоторую ручную работу с заголовками.

Я также использую библиотеку OkHttp 3 для сетевого подключения, но вы, вероятно, могли бы адаптировать мой код для других библиотек.

Основная идея заключается в том, что вам нужно договориться с сервером о подключении.

Шаг 1:

Когда вы попытаетесь подключиться в первый раз, у вас не получится и вы получите некоторую информацию в заголовке:

WWW-Аутентификация: переговоры

WWW-Аутентификация: NTLM

Шаг 2:

Вам необходимо сгенерировать ключ типа 1 (с необязательными параметрами домена и рабочей станции), используя библиотеку jcifs, и попытаться подключиться снова. Вы снова потерпите неудачу, но получите некоторую полезную информацию в шапке:

WWW-Аутентификация: NTLM very_long_challenge_key

Шаг 3:

Вам необходимо сгенерировать ключ типа 3 с этим ключом вызова + логин + пароль, используя библиотеку jcifs. Тогда соединение будет успешным!

Теперь немного кода, добавьте зависимость к библиотекам в файле build.gradle вашего приложения:

compile files('libs/jcifs-1.3.18.jar')
compile 'com.squareup.okhttp3:okhttp:3.4.1'

Банку можно найти здесь: https://jcifs.samba.org/src/

Тогда класс NTLMAuthenticator

import android.support.annotation.NonNull;

import java.io.IOException;
import java.util.List;

import jcifs.ntlmssp.NtlmFlags;
import jcifs.ntlmssp.Type1Message;
import jcifs.ntlmssp.Type2Message;
import jcifs.ntlmssp.Type3Message;
import jcifs.util.Base64;
import okhttp3.Authenticator;
import okhttp3.Credentials;
import okhttp3.Request;
import okhttp3.Response;
import okhttp3.Route;

/**
 * Created by Arnaud Guyon on 07.02.17.
 */

public class NTLMAuthenticator implements Authenticator {

    private static final int TYPE_1_FLAGS =
            NtlmFlags.NTLMSSP_NEGOTIATE_56 |
                    NtlmFlags.NTLMSSP_NEGOTIATE_128 |
                    NtlmFlags.NTLMSSP_NEGOTIATE_NTLM2 |
                    NtlmFlags.NTLMSSP_NEGOTIATE_ALWAYS_SIGN |
                    NtlmFlags.NTLMSSP_REQUEST_TARGET;

    private String mLogin;
    private String mPassword;
    private String mDomain;
    private String mWorkstation;

    public NTLMAuthenticator(@NonNull String login, @NonNull String password) {
        this(login, password, "", "");
    }

    public NTLMAuthenticator(@NonNull String login, @NonNull String password, @NonNull String domain, @NonNull String workstation) {
        mLogin = login;
        mPassword = password;
        mDomain = domain;
        mWorkstation = workstation;
    }

    @Override
    public Request authenticate(Route route, Response response) throws IOException {

        List<String> authHeaders = response.headers("WWW-Authenticate");
        if (authHeaders != null) {
            boolean negociate = false;
            boolean ntlm = false;
            String ntlmValue = null;
            for (String authHeader : authHeaders) {
                if (authHeader.equalsIgnoreCase("Negotiate")) {
                    negociate = true;
                }
                if (authHeader.equalsIgnoreCase("NTLM")) {
                    ntlm = true;
                }
                if (authHeader.startsWith("NTLM ")) {
                    ntlmValue = authHeader.substring(5);
                }
            }

            if (negociate && ntlm) {
                String type1Msg = generateType1Msg(mDomain, mWorkstation);
                String header = "NTLM " + type1Msg;
                return response.request().newBuilder().header("Authorization", header).build();
            } else if (ntlmValue != null) {
                String type3Msg = generateType3Msg(mLogin, mPassword, mDomain, mWorkstation, ntlmValue);
                String ntlmHeader = "NTLM " + type3Msg;
                return response.request().newBuilder().header("Authorization", ntlmHeader).build();
            }
        }

        if (responseCount(response) <= 3) {
            String credential = Credentials.basic(mLogin, mPassword);
            return response.request().newBuilder().header("Authorization", credential).build();
        }

        return null;
    }

    private String generateType1Msg(@NonNull String domain, @NonNull String workstation) {
        final Type1Message type1Message = new Type1Message(TYPE_1_FLAGS, domain, workstation);
        byte[] source = type1Message.toByteArray();
        return Base64.encode(source);
    }

    private String generateType3Msg(final String login, final String password, final String domain, final String workstation, final String challenge) {
        Type2Message type2Message;
        try {
            byte[] decoded = Base64.decode(challenge);
            type2Message = new Type2Message(decoded);
        } catch (final IOException exception) {
            exception.printStackTrace();
            return null;
        }
        final int type2Flags = type2Message.getFlags();
        final int type3Flags = type2Flags
                & (0xffffffff ^ (NtlmFlags.NTLMSSP_TARGET_TYPE_DOMAIN | NtlmFlags.NTLMSSP_TARGET_TYPE_SERVER));
        final Type3Message type3Message = new Type3Message(type2Message, password, domain,
                login, workstation, type3Flags);
        return Base64.encode(type3Message.toByteArray());
    }

    private int responseCount(Response response) {
        int result = 1;
        while ((response = response.priorResponse()) != null) {
            result++;
        }
        return result;
    }

}

Затем, когда вы создаете ваш OkHttpClient, добавьте этот аутентификатор:

OkHttpClient okHttpClient = new OkHttpClient.Builder()
        .authenticator(new NTLMAuthenticator(login, password))
        // .some other init here if necessary
        .build();

А затем выполняйте ваши запросы как обычно.

0 голосов
/ 04 октября 2013
...