Ошибка аутентификации Kerberos «Запрос - это повтор 34» - PullRequest
0 голосов
/ 07 января 2019

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

package com.onenew.http.kerberos.test;
import java.io.IOException;
import java.security.Principal;
import java.security.PrivilegedActionException;
import java.security.PrivilegedExceptionAction;
import java.util.Set;
import javax.security.auth.Subject;
import javax.security.auth.login.LoginContext;
import javax.security.auth.login.LoginException;
import org.apache.commons.codec.binary.Base64;
import org.ietf.jgss.GSSContext;
import org.ietf.jgss.GSSCredential;
import org.ietf.jgss.GSSException;
import org.ietf.jgss.GSSManager;
import org.ietf.jgss.GSSName;
import org.ietf.jgss.Oid;

public class App {
    private static final String loginContextName = "xyz";
    private static final String userName = "username@EXAMPLE.COM";
    private static final String targetServerSpn = "server/EXAMPLE1.COM@EXAMPLE.COM";

    public static void main(String[] args) {
        try {
            System.setProperty("java.security.auth.login.config", "/var/jaas.conf");
            Oid krb5MechOid = new Oid("1.2.840.113554.1.2.2");
            Oid krb5PrincipalNameOid = new Oid("1.2.840.113554.1.2.2.1");
            GSSManager manager = GSSManager.getInstance();
            LoginContext lc = new LoginContext(loginContextName);
            lc.login();
            Subject subject = lc.getSubject();
            GSSCredential clientGssCreds = (GSSCredential) Subject.doAs(subject,
                    new CreateClientGSSCredentialAction(krb5MechOid, krb5PrincipalNameOid, manager));

            // create target server SPN
            GSSName gssServerName = manager.createName(targetServerSpn, krb5PrincipalNameOid);

            GSSContext clientContext = manager.createContext(gssServerName, krb5MechOid,
                    clientGssCreds, GSSContext.DEFAULT_LIFETIME);
            // optional enable GSS credential delegation
            clientContext.requestCredDeleg(true);

            byte[] spnegoToken = new byte[0];

            // create a SPNEGO token for the target server
            spnegoToken = clientContext.initSecContext(spnegoToken, 0, spnegoToken.length);
            String token = Base64.encodeBase64String(spnegoToken);
            System.out.println("Base64 Token:"+token);
            clientContext.dispose();

        }catch (GSSException e) {
            System.out.println("GSSException");
            System.out.println(e.getMessage());
        } catch (PrivilegedActionException e) {
            System.out.println("PrivilegedActionException");
            System.out.println(e.getMessage());
        } catch (LoginException e) {
            System.out.println("LoginException");
            System.out.println(e.getMessage());
        }

    }

    private static final class CreateClientGSSCredentialAction implements PrivilegedExceptionAction<GSSCredential> {
        private Oid krb5MechOid;
        private Oid krb5PrincipalNameOid;
        private GSSManager manager;

        private CreateClientGSSCredentialAction(Oid krb5MechOid, Oid krb5PrincipalNameOid, GSSManager manager) {
            this.krb5MechOid = krb5MechOid;
            this.krb5PrincipalNameOid = krb5PrincipalNameOid;
            this.manager = manager;
        }

        public GSSCredential run() throws GSSException, Exception {
            try {
                GSSName gssClientName = manager.createName(userName, krb5PrincipalNameOid);
                GSSCredential gssClientCred = manager.createCredential(gssClientName,
                        GSSCredential.DEFAULT_LIFETIME, krb5MechOid, GSSCredential.INITIATE_ONLY);
                return gssClientCred;
            } catch (GSSException e) {
                System.out.println("run method GSSException");
                System.out.println(e.getMessage());
            } catch (Exception e) {
                System.out.println("run method Exception");
                System.out.println(e.getMessage());
            }
            return null;
        }
    }
}

Затем я использую сгенерированный токен для запроса службы REST, созданной Jetty и Jersey. В службе REST есть KerberosFilter, который реализуется из ContainerRequestFilter, ниже приведен код:

package com.onenew.rest.security.filter;

import javax.security.auth.Subject;
import javax.security.auth.login.LoginContext;
import javax.security.auth.login.LoginException;
import javax.ws.rs.container.ContainerRequestContext;
import javax.ws.rs.container.ContainerRequestFilter;
import javax.ws.rs.core.HttpHeaders;
import java.io.IOException;
import java.security.Principal;
import java.security.PrivilegedActionException;
import java.security.PrivilegedExceptionAction;
import java.util.Set;

import javax.ws.rs.core.Response;
import io.confluent.kafka.schemaregistry.rest.SchemaRegistryConfig;
import io.confluent.kafka.schemaregistry.storage.SchemaRegistry;
import io.confluent.rest.RestConfigException;
import org.apache.cxf.common.security.SimplePrincipal;
import org.apache.cxf.common.security.SimpleSecurityContext;
import org.apache.cxf.common.util.Base64Exception;
import org.apache.cxf.common.util.Base64Utility;
import org.ietf.jgss.GSSContext;
import org.ietf.jgss.GSSException;
import org.ietf.jgss.GSSManager;
import org.ietf.jgss.GSSName;
import org.ietf.jgss.Oid;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public final class KafkaKerberosFilter implements ContainerRequestFilter {
    private static final Logger log = LoggerFactory.getLogger(KafkaKerberosFilter.class);

    private static String REALM="EXAMPLE.COM";
    private static final String AUTHENTICATION_SCHEME = "Negotiate";
    private static final String KERBEROS_OID = "1.2.840.113554.1.2.2";
    private static final String KERBEROS_PRINCIPAL_OID="1.2.840.113554.1.2.2.1";

    private String loginContextName = "xyzService";
    private String servicePrincipalName;

    public KafkaKerberosFilter() {

    }

    public void filter(ContainerRequestContext requestContext) throws IOException {
        // Get the Authorization header from the request
        String authorizationHeader = requestContext.getHeaderString(HttpHeaders.AUTHORIZATION);
        if(authorizationHeader==null||authorizationHeader.length()==0){
            String errorMsg="invalid authorization header";
            log.error(errorMsg);
            abortWithUnauthorized(requestContext,errorMsg);
            return;
        }
        String[] authPair = authorizationHeader.split(" ");
        if (authPair.length != 2 || !AUTHENTICATION_SCHEME.equalsIgnoreCase(authPair[0])) {
            String errorMsg="Negotiate Authorization scheme is expected";
            log.error(errorMsg);
            abortWithUnauthorized(requestContext,errorMsg);
            return;
        }
        log.debug("Base64 Token:"+authPair[1]);
        byte[] serviceTicket;
        try{
            serviceTicket = Base64Utility.decode(authPair[1]);
            log.debug("Decoded Token:"+serviceTicket.toString());
        }catch(Base64Exception e){
            String errorMsg="can't decode kerberos ticket";
            log.error(errorMsg+" "+authPair[1]);
            abortWithUnauthorized(requestContext,errorMsg);
            return;
        }
        try{
            Subject serviceSubject = loginAndGetSubject();
            GSSContext gssContext = createGSSContext(requestContext);
            Subject.doAs(serviceSubject, new ValidateServiceTicketAction(gssContext, serviceTicket));
            if (!gssContext.getCredDelegState()) {
                gssContext.dispose();
                gssContext = null;
            }

        }catch(LoginException e){
            String errorMsg="LoginContext can not be initialized";
            log.error(errorMsg+" "+loginContextName);
            abortWithUnauthorized(requestContext,errorMsg);
            return;
        }catch (GSSException e) {
            String errorMsg="GSS API exception";
            log.error(errorMsg+" "+e.getMessage());
            abortWithUnauthorized(requestContext,errorMsg);
            return;
        } catch (PrivilegedActionException e) {
            String errorMsg="PrivilegedActionException";
            log.error(errorMsg+" "+e.getMessage());
            abortWithUnauthorized(requestContext,errorMsg);
            return;
        }
    }

    private void abortWithUnauthorized(ContainerRequestContext requestContext,String error) {

        // Abort the filter chain with a 401 status code response
        // The WWW-Authenticate header is sent along with the response
        requestContext.abortWith(Response.status(Response.Status.UNAUTHORIZED)
                .header(HttpHeaders.WWW_AUTHENTICATE, AUTHENTICATION_SCHEME + " realm=\"" + REALM + "\",\nerror_description=\""+error+"\"").build());
    }

    protected Subject loginAndGetSubject() throws LoginException {

        // The login without a callback can work if
        // - Kerberos keytabs are used with a principal name set in the JAAS config
        // - Kerberos is integrated into the OS logon process
        //   meaning that a process which runs this code has the
        //   user identity
        LoginContext lc = new LoginContext(loginContextName);
        lc.login();
        Subject subject = lc.getSubject();
        return subject;
    }

    protected GSSContext createGSSContext(ContainerRequestContext requestContext) throws GSSException {
        Oid principalOid = new Oid(KERBEROS_PRINCIPAL_OID);
        Oid oid = new Oid(KERBEROS_OID);
        GSSManager gssManager = GSSManager.getInstance();
        String spn = getCompleteServicePrincipalName(requestContext);
        GSSName gssService = gssManager.createName(spn,principalOid);
        return gssManager.createContext(gssService,
                   oid, null, GSSContext.DEFAULT_LIFETIME);
    }

    protected String getCompleteServicePrincipalName(ContainerRequestContext requestContext) {
        String name = servicePrincipalName == null
            ? "server/" + requestContext.getUriInfo().getBaseUri().getHost() : servicePrincipalName;
            name += "@" + REALM;
        log.debug("Service Principal Name:"+name);
        return name;
    }

    public void setServicePrincipalName(String servicePrincipalName) {
        this.servicePrincipalName = servicePrincipalName;
    }

    private static final class ValidateServiceTicketAction implements PrivilegedExceptionAction<byte[]> {
        private final GSSContext context;
        private final byte[] token;

        private ValidateServiceTicketAction(GSSContext context, byte[] token) {
            this.context = context;
            this.token = token;
        }

        public byte[] run() throws GSSException {
            try{
                return context.acceptSecContext(token, 0, token.length);
            }catch(GSSException e){
                log.error("PrivilegedExceptionAction run error:"+e.getMessage());
                throw e;
            }
        }
    }
}

Первый запрос с сгенерированным токеном - успех. Но следующие запросы с одним и тем же токеном не были выполнены. Выполнение метода класса ValidateServiceTicketAction вызывает GSSException, что говорит о том, что «Не указан сбой на уровне GSS-API (уровень механизма: запрос является воспроизведением (34))». Я гуглил и не могу найти решение, большинство ответов говорят, что время должно быть синхронизировано, но я уже все время синхронизирую на клиенте, службе REST и KDC.

...