jdk/src/share/classes/sun/security/jgss/spnego/SpNegoMechFactory.java
author weijun
Fri, 27 Nov 2009 08:51:58 +0800
changeset 4338 f36521ae16db
parent 2942 37d9baeb7518
child 5506 202f599c92aa
permissions -rw-r--r--
6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism Reviewed-by: valeriep
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
2942
37d9baeb7518 6578647: Undefined requesting URL in java.net.Authenticator.getPasswordAuthentication()
weijun
parents: 2
diff changeset
     2
 * Copyright 2005-2009 Sun Microsystems, Inc.  All Rights Reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
90ce3da70b43 Initial load
duke
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Sun designates this
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
90ce3da70b43 Initial load
duke
parents:
diff changeset
     9
 * by Sun in the LICENSE file that accompanied this code.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    21
 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    22
 * CA 95054 USA or visit www.sun.com if you need additional information or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    23
 * have any questions.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
package sun.security.jgss.spnego;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
import org.ietf.jgss.*;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    29
import sun.security.jgss.*;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    30
import sun.security.jgss.spi.*;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    31
import sun.security.jgss.krb5.Krb5MechFactory;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    32
import sun.security.jgss.krb5.Krb5InitCredential;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    33
import sun.security.jgss.krb5.Krb5AcceptCredential;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    34
import sun.security.jgss.krb5.Krb5NameElement;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    35
import java.security.Provider;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    36
import java.util.Vector;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    37
90ce3da70b43 Initial load
duke
parents:
diff changeset
    38
/**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
 * SpNego Mechanism plug in for JGSS
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
 * This is the properties object required by the JGSS framework.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    41
 * All mechanism specific information is defined here.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    43
 * @author Seema Malkani
90ce3da70b43 Initial load
duke
parents:
diff changeset
    44
 * @since 1.6
90ce3da70b43 Initial load
duke
parents:
diff changeset
    45
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    46
90ce3da70b43 Initial load
duke
parents:
diff changeset
    47
public final class SpNegoMechFactory implements MechanismFactory {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    48
90ce3da70b43 Initial load
duke
parents:
diff changeset
    49
    static final Provider PROVIDER =
90ce3da70b43 Initial load
duke
parents:
diff changeset
    50
        new sun.security.jgss.SunProvider();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    51
90ce3da70b43 Initial load
duke
parents:
diff changeset
    52
    static final Oid GSS_SPNEGO_MECH_OID =
90ce3da70b43 Initial load
duke
parents:
diff changeset
    53
        GSSUtil.createOid("1.3.6.1.5.5.2");
90ce3da70b43 Initial load
duke
parents:
diff changeset
    54
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
    private static Oid[] nameTypes =
90ce3da70b43 Initial load
duke
parents:
diff changeset
    56
        new Oid[] { GSSName.NT_USER_NAME,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    57
                        GSSName.NT_HOSTBASED_SERVICE,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
                        GSSName.NT_EXPORT_NAME};
90ce3da70b43 Initial load
duke
parents:
diff changeset
    59
4338
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
    60
    // The default underlying mech of SPNEGO, must not be SPNEGO itself.
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
    61
    private static final Oid DEFAULT_SPNEGO_MECH_OID =
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
    62
            ProviderList.DEFAULT_MECH_OID.equals(GSS_SPNEGO_MECH_OID)?
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
    63
                GSSUtil.GSS_KRB5_MECH_OID:
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
    64
                ProviderList.DEFAULT_MECH_OID;
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
    65
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    66
    // Use an instance of a GSSManager whose provider list
90ce3da70b43 Initial load
duke
parents:
diff changeset
    67
    // does not include native provider
90ce3da70b43 Initial load
duke
parents:
diff changeset
    68
    final GSSManagerImpl manager;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    69
    final Oid[] availableMechs;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    70
90ce3da70b43 Initial load
duke
parents:
diff changeset
    71
    private static SpNegoCredElement getCredFromSubject(GSSNameSpi name,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    72
                                                        boolean initiate)
90ce3da70b43 Initial load
duke
parents:
diff changeset
    73
        throws GSSException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    74
        Vector<SpNegoCredElement> creds =
90ce3da70b43 Initial load
duke
parents:
diff changeset
    75
            GSSUtil.searchSubject(name, GSS_SPNEGO_MECH_OID,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    76
                initiate, SpNegoCredElement.class);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
90ce3da70b43 Initial load
duke
parents:
diff changeset
    78
        SpNegoCredElement result = ((creds == null || creds.isEmpty()) ?
90ce3da70b43 Initial load
duke
parents:
diff changeset
    79
                                    null : creds.firstElement());
90ce3da70b43 Initial load
duke
parents:
diff changeset
    80
90ce3da70b43 Initial load
duke
parents:
diff changeset
    81
        // Force permission check before returning the cred to caller
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
        if (result != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    83
            GSSCredentialSpi cred = result.getInternalCred();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    84
            if (GSSUtil.isKerberosMech(cred.getMechanism())) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    85
                if (initiate) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    86
                    Krb5InitCredential krbCred = (Krb5InitCredential) cred;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    87
                    Krb5MechFactory.checkInitCredPermission
90ce3da70b43 Initial load
duke
parents:
diff changeset
    88
                        ((Krb5NameElement) krbCred.getName());
90ce3da70b43 Initial load
duke
parents:
diff changeset
    89
                } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    90
                    Krb5AcceptCredential krbCred = (Krb5AcceptCredential) cred;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    91
                    Krb5MechFactory.checkAcceptCredPermission
90ce3da70b43 Initial load
duke
parents:
diff changeset
    92
                        ((Krb5NameElement) krbCred.getName(), name);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    93
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    94
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    95
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    96
        return result;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    97
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    98
2942
37d9baeb7518 6578647: Undefined requesting URL in java.net.Authenticator.getPasswordAuthentication()
weijun
parents: 2
diff changeset
    99
    public SpNegoMechFactory(GSSCaller caller) {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   100
        manager = new GSSManagerImpl(caller, false);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   101
        Oid[] mechs = manager.getMechs();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   102
        availableMechs = new Oid[mechs.length-1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   103
        for (int i = 0, j = 0; i < mechs.length; i++) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   104
            // Skip SpNego mechanism
90ce3da70b43 Initial load
duke
parents:
diff changeset
   105
            if (!mechs[i].equals(GSS_SPNEGO_MECH_OID)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   106
                availableMechs[j++] = mechs[i];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   107
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   108
        }
4338
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   109
        // Move the preferred mech to first place
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   110
        for (int i=0; i<availableMechs.length; i++) {
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   111
            if (availableMechs[i].equals(DEFAULT_SPNEGO_MECH_OID)) {
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   112
                if (i != 0) {
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   113
                    availableMechs[i] = availableMechs[0];
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   114
                    availableMechs[0] = DEFAULT_SPNEGO_MECH_OID;
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   115
                }
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   116
                break;
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   117
            }
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   118
        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   119
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   120
90ce3da70b43 Initial load
duke
parents:
diff changeset
   121
    public GSSNameSpi getNameElement(String nameStr, Oid nameType)
4338
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   122
            throws GSSException {
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   123
        return manager.getNameElement(
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   124
                nameStr, nameType, DEFAULT_SPNEGO_MECH_OID);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   125
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   126
90ce3da70b43 Initial load
duke
parents:
diff changeset
   127
    public GSSNameSpi getNameElement(byte[] name, Oid nameType)
4338
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   128
            throws GSSException {
f36521ae16db 6770883: Infinite loop if SPNEGO specified as sun.security.jgss.mechanism
weijun
parents: 2942
diff changeset
   129
        return manager.getNameElement(name, nameType, DEFAULT_SPNEGO_MECH_OID);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   130
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   131
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
    public GSSCredentialSpi getCredentialElement(GSSNameSpi name,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   133
           int initLifetime, int acceptLifetime,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
           int usage) throws GSSException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
90ce3da70b43 Initial load
duke
parents:
diff changeset
   136
        SpNegoCredElement credElement = getCredFromSubject
90ce3da70b43 Initial load
duke
parents:
diff changeset
   137
            (name, (usage != GSSCredential.ACCEPT_ONLY));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   138
90ce3da70b43 Initial load
duke
parents:
diff changeset
   139
        if (credElement == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
            // get CredElement for the default Mechanism
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
            credElement = new SpNegoCredElement
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
                (manager.getCredentialElement(name, initLifetime,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
                acceptLifetime, null, usage));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
        return credElement;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   146
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
90ce3da70b43 Initial load
duke
parents:
diff changeset
   148
    public GSSContextSpi getMechanismContext(GSSNameSpi peer,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
                             GSSCredentialSpi myInitiatorCred, int lifetime)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   150
        throws GSSException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   151
        // get SpNego mechanism context
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
        if (myInitiatorCred == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   153
            myInitiatorCred = getCredFromSubject(null, true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   154
        } else if (!(myInitiatorCred instanceof SpNegoCredElement)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   155
            // convert to SpNegoCredElement
90ce3da70b43 Initial load
duke
parents:
diff changeset
   156
            SpNegoCredElement cred = new SpNegoCredElement(myInitiatorCred);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   157
            return new SpNegoContext(this, peer, cred, lifetime);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   158
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   159
        return new SpNegoContext(this, peer, myInitiatorCred, lifetime);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   160
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
90ce3da70b43 Initial load
duke
parents:
diff changeset
   162
    public GSSContextSpi getMechanismContext(GSSCredentialSpi myAcceptorCred)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   163
        throws GSSException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   164
        // get SpNego mechanism context
90ce3da70b43 Initial load
duke
parents:
diff changeset
   165
        if (myAcceptorCred == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   166
            myAcceptorCred = getCredFromSubject(null, false);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   167
        } else if (!(myAcceptorCred instanceof SpNegoCredElement)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   168
            // convert to SpNegoCredElement
90ce3da70b43 Initial load
duke
parents:
diff changeset
   169
            SpNegoCredElement cred = new SpNegoCredElement(myAcceptorCred);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   170
            return new SpNegoContext(this, cred);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
        return new SpNegoContext(this, myAcceptorCred);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
    public GSSContextSpi getMechanismContext(byte[] exportedContext)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
        throws GSSException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
        // get SpNego mechanism context
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
        return new SpNegoContext(this, exportedContext);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
    public final Oid getMechanismOid() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
        return GSS_SPNEGO_MECH_OID;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
    public Provider getProvider() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
        return PROVIDER;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
    public Oid[] getNameTypes() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
        // nameTypes is cloned in GSSManager.getNamesForMech
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
        return nameTypes;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
}