/* * Copyright (C) 2019 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.net.eap; import android.annotation.IntDef; import android.annotation.NonNull; import android.annotation.Nullable; import android.annotation.SystemApi; import android.os.PersistableBundle; import android.telephony.Annotation.UiccAppType; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.net.ipsec.ike.utils.IkeCertUtils; import com.android.server.vcn.util.PersistableBundleUtils; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.security.cert.CertificateEncodingException; import java.security.cert.TrustAnchor; import java.security.cert.X509Certificate; import java.util.Arrays; import java.util.Collections; import java.util.HashMap; import java.util.Map; import java.util.Objects; /** * EapSessionConfig represents a container for EAP method configuration. * *
The EAP authentication server decides which EAP method is used, so clients are encouraged to
* provide configs for several EAP methods.
*/
public final class EapSessionConfig {
private static final String EAP_ID_KEY = "eapIdentity";
private static final String EAP_METHOD_CONFIGS_KEY = "eapConfigs";
private static final byte[] DEFAULT_IDENTITY = new byte[0];
// IANA -> EapMethodConfig for that method
private final Map Constructed EapSessionConfigs are guaranteed to be valid, as checked by the
* EapSessionConfig.Builder
*
* @hide
*/
@NonNull
public static EapSessionConfig fromPersistableBundle(@NonNull PersistableBundle in) {
Objects.requireNonNull(in, "PersistableBundle is null");
EapSessionConfig.Builder builder = new EapSessionConfig.Builder();
PersistableBundle eapIdBundle = in.getPersistableBundle(EAP_ID_KEY);
Objects.requireNonNull(eapIdBundle, "EAP ID bundle is null");
byte[] eapId = PersistableBundleUtils.toByteArray(eapIdBundle);
builder.setEapIdentity(eapId);
PersistableBundle configsBundle = in.getPersistableBundle(EAP_METHOD_CONFIGS_KEY);
Objects.requireNonNull(configsBundle, "EAP method configs bundle is null");
Map Tunneled EAP-TTLS authentications are disallowed, as running multiple layers of
* EAP-TTLS increases the data footprint but has no discernible benefits over a single
* EAP-TTLS session with a non EAP-TTLS method nested inside it.
*
* @param serverCaCert the CA certificate for validating the received server certificate(s).
* If a certificate is provided, it MUST be the root CA used by the server, or
* authentication will fail. If no certificate is provided, any root CA in the system's
* truststore is considered acceptable.
* @param innerEapSessionConfig represents the configuration for the inner EAP instance
* @return Builder this, to facilitate chaining
*/
@NonNull
public Builder setEapTtlsConfig(
@Nullable X509Certificate serverCaCert,
@NonNull EapSessionConfig innerEapSessionConfig) {
mEapConfigs.put(
EapMethodConfig.EAP_TYPE_TTLS,
new EapTtlsConfig(serverCaCert, innerEapSessionConfig));
return this;
}
/**
* Adds an EAP method configuration. Internal use only.
*
* This method will override the previously set configuration with the same method type.
*
* @hide
*/
@NonNull
public Builder addEapMethodConfig(@NonNull EapMethodConfig config) {
Objects.requireNonNull(config, "EapMethodConfig is null");
mEapConfigs.put(config.mMethodType, config);
return this;
}
/**
* Constructs and returns an EapSessionConfig with the configurations applied to this
* Builder.
*
* @return the EapSessionConfig constructed by this Builder.
*/
@NonNull
public EapSessionConfig build() {
if (mEapConfigs.isEmpty()) {
throw new IllegalStateException("Must have at least one EAP method configured");
}
return new EapSessionConfig(mEapConfigs, mEapIdentity);
}
}
/** EapMethodConfig represents a generic EAP method configuration. */
public abstract static class EapMethodConfig {
private static final String METHOD_TYPE = "methodType";
/** @hide */
@Retention(RetentionPolicy.SOURCE)
@IntDef({EAP_TYPE_SIM, EAP_TYPE_TTLS, EAP_TYPE_AKA, EAP_TYPE_MSCHAP_V2, EAP_TYPE_AKA_PRIME})
public @interface EapMethod {}
// EAP Type values defined by IANA
// @see https://www.iana.org/assignments/eap-numbers/eap-numbers.xhtml
/**
* EAP-Type value for the EAP-SIM method.
*
* To include EAP-SIM as an authentication method, see {@link
* EapSessionConfig.Builder#setEapSimConfig(int, int)}.
*
* @see RFC 4186, Extensible Authentication
* Protocol Method for Global System for Mobile Communications (GSM) Subscriber Identity
* Modules (EAP-SIM)
*/
public static final int EAP_TYPE_SIM = 18;
/**
* EAP-Type value for the EAP-TTLS method.
*
* To include EAP-TTLS as an authentication method, see {@link
* EapSessionConfig.Builder#setEapTtlsConfig(X509Certificate, EapSessionConfig)}.
*
* @see RFC 5281, Extensible Authentication
* Protocol Tunneled Transport Layer Security Authenticated Protocol Version 0
* (EAP-TTLSv0)
*/
public static final int EAP_TYPE_TTLS = 21;
/**
* EAP-Type value for the EAP-AKA method.
*
* To include EAP-AKA as an authentication method, see {@link
* EapSessionConfig.Builder#setEapAkaConfig(int, int)}.
*
* @see RFC 4187, Extensible Authentication
* Protocol Method for 3rd Generation Authentication and Key Agreement (EAP-AKA)
*/
public static final int EAP_TYPE_AKA = 23;
/**
* EAP-Type value for the EAP-MS-CHAPv2 method.
*
* To include EAP-MS-CHAPv2 as an authentication method, see {@link
* EapSessionConfig.Builder#setEapMsChapV2Config(String, String)}.
*
* @see Microsoft
* EAP CHAP Extensions Draft (EAP MSCHAPv2)
*/
public static final int EAP_TYPE_MSCHAP_V2 = 26;
/**
* EAP-Type value for the EAP-AKA' method.
*
* To include EAP-AKA' as an authentication method, see {@link
* EapSessionConfig.Builder#setEapAkaPrimeConfig(int, int, String, boolean)}.
*
* @see RFC 5448, Improved Extensible
* Authentication Protocol Method for 3rd Generation Authentication and Key Agreement
* (EAP-AKA')
*/
public static final int EAP_TYPE_AKA_PRIME = 50;
@EapMethod private final int mMethodType;
/** @hide */
EapMethodConfig(@EapMethod int methodType) {
mMethodType = methodType;
}
/**
* Constructs this object by deserializing a PersistableBundle
*
* @hide
*/
@NonNull
public static EapMethodConfig fromPersistableBundle(PersistableBundle in) {
Objects.requireNonNull(in, "PersistableBundle is null");
int methodType = in.getInt(METHOD_TYPE);
switch (methodType) {
case EAP_TYPE_SIM:
return EapSimConfig.fromPersistableBundle(in);
case EAP_TYPE_AKA:
return EapAkaConfig.fromPersistableBundle(in);
case EAP_TYPE_AKA_PRIME:
return EapAkaPrimeConfig.fromPersistableBundle(in);
case EAP_TYPE_MSCHAP_V2:
return EapMsChapV2Config.fromPersistableBundle(in);
case EAP_TYPE_TTLS:
return EapTtlsConfig.fromPersistableBundle(in);
default:
throw new IllegalArgumentException("Invalid EAP Type: " + methodType);
}
}
/**
* Serializes this object to a PersistableBundle
*
* @hide
*/
@NonNull
protected PersistableBundle toPersistableBundle() {
final PersistableBundle result = new PersistableBundle();
result.putInt(METHOD_TYPE, mMethodType);
return result;
}
/**
* Retrieves the EAP method type
*
* @return the IANA-defined EAP method constant
*/
public int getMethodType() {
return mMethodType;
}
/**
* Check if this is EAP-only safe method.
*
* @return whether the method is EAP-only safe
*
* @see RFC 5998#section 4, for safe eap
* methods
*
* @hide
*/
public boolean isEapOnlySafeMethod() {
return false;
}
/** @hide */
@Override
public int hashCode() {
return Objects.hash(mMethodType);
}
/** @hide */
@Override
public boolean equals(Object o) {
if (!(o instanceof EapMethodConfig)) {
return false;
}
return mMethodType == ((EapMethodConfig) o).mMethodType;
}
}
/**
* EapUiccConfig represents the configs needed for EAP methods that rely on UICC cards for
* authentication.
*
* @hide
* @deprecated This class is not useful. Callers should only use its two subclasses {@link
* EapSimConfig} and {@link EapAkaConfig}
*/
@Deprecated
@SystemApi
public abstract static class EapUiccConfig extends EapMethodConfig {
/** @hide */
protected static final String SUB_ID_KEY = "subId";
/** @hide */
protected static final String APP_TYPE_KEY = "apptype";
private final int mSubId;
private final int mApptype;
private EapUiccConfig(@EapMethod int methodType, int subId, @UiccAppType int apptype) {
super(methodType);
mSubId = subId;
mApptype = apptype;
}
/**
* Serializes this object to a PersistableBundle
*
* @hide
*/
@Override
@NonNull
protected PersistableBundle toPersistableBundle() {
final PersistableBundle result = super.toPersistableBundle();
result.putInt(SUB_ID_KEY, mSubId);
result.putInt(APP_TYPE_KEY, mApptype);
return result;
}
/**
* Retrieves the subId
*
* @return the subId
*/
public int getSubId() {
return mSubId;
}
/**
* Retrieves the UICC app type
*
* return the type
*/
public @UiccAppType int getAppType() {
return mApptype;
}
/** @hide */
@Override
public boolean isEapOnlySafeMethod() {
return true;
}
/** @hide */
@Override
public int hashCode() {
return Objects.hash(super.hashCode(), mSubId, mApptype);
}
/** @hide */
@Override
public boolean equals(Object o) {
if (!super.equals(o) || !(o instanceof EapUiccConfig)) {
return false;
}
EapUiccConfig other = (EapUiccConfig) o;
return mSubId == other.mSubId && mApptype == other.mApptype;
}
}
/**
* EapSimConfig represents the configs needed for an EAP SIM session.
*/
public static class EapSimConfig extends EapUiccConfig {
/** @hide */
@VisibleForTesting
public EapSimConfig(int subId, @UiccAppType int apptype) {
super(EAP_TYPE_SIM, subId, apptype);
}
/**
* Constructs this object by deserializing a PersistableBundle
*
* @hide
*/
@NonNull
public static EapSimConfig fromPersistableBundle(@NonNull PersistableBundle in) {
Objects.requireNonNull(in, "PersistableBundle is null");
return new EapSimConfig(in.getInt(SUB_ID_KEY), in.getInt(APP_TYPE_KEY));
}
}
/**
* EapAkaConfig represents the configs needed for an EAP AKA session.
*/
public static class EapAkaConfig extends EapUiccConfig {
private static final String AKA_OPTION_KEY = "akaOption";
private final EapAkaOption mEapAkaOption;
/** @hide */
@VisibleForTesting
public EapAkaConfig(int subId, @UiccAppType int apptype) {
this(EAP_TYPE_AKA, subId, apptype, null);
}
/** @hide */
@VisibleForTesting
public EapAkaConfig(int subId, @UiccAppType int apptype, EapAkaOption options) {
this(EAP_TYPE_AKA, subId, apptype, options);
}
/** @hide */
EapAkaConfig(int methodType, int subId, @UiccAppType int apptype, EapAkaOption options) {
super(methodType, subId, apptype);
mEapAkaOption = options;
}
/**
* Constructs this object by deserializing a PersistableBundle
*
* @hide
*/
@NonNull
public static EapAkaConfig fromPersistableBundle(@NonNull PersistableBundle in) {
Objects.requireNonNull(in, "PersistableBundle is null");
EapAkaOption eapAkaOption = null;
PersistableBundle bundle = in.getPersistableBundle(AKA_OPTION_KEY);
if (bundle != null) {
eapAkaOption = EapAkaOption.fromPersistableBundle(bundle);
}
return new EapAkaConfig(in.getInt(SUB_ID_KEY), in.getInt(APP_TYPE_KEY), eapAkaOption);
}
/**
* Serializes this object to a PersistableBundle
*
* @hide
*/
@Override
@NonNull
protected PersistableBundle toPersistableBundle() {
final PersistableBundle result = super.toPersistableBundle();
if (mEapAkaOption != null) {
result.putPersistableBundle(AKA_OPTION_KEY, mEapAkaOption.toPersistableBundle());
}
return result;
}
/**
* Retrieves EapAkaOption
*
* @return the {@link EapAkaOption}
*/
@NonNull
public EapAkaOption getEapAkaOption() {
return mEapAkaOption;
}
}
/**
* EapAkaPrimeConfig represents the configs needed for an EAP-AKA' session.
*/
public static class EapAkaPrimeConfig extends EapAkaConfig {
private static final String NETWORK_NAME_KEY = "networkName";
private static final String ALL_MISMATCHED_NETWORK_KEY = "allowMismatchedNetworkNames";
@NonNull private final String mNetworkName;
private final boolean mAllowMismatchedNetworkNames;
/** @hide */
@VisibleForTesting
public EapAkaPrimeConfig(
int subId,
@UiccAppType int apptype,
@NonNull String networkName,
boolean allowMismatchedNetworkNames) {
super(EAP_TYPE_AKA_PRIME, subId, apptype, null);
Objects.requireNonNull(networkName, "networkName must not be null");
mNetworkName = networkName;
mAllowMismatchedNetworkNames = allowMismatchedNetworkNames;
}
/**
* Constructs this object by deserializing a PersistableBundle
*
* @hide
*/
@NonNull
public static EapAkaPrimeConfig fromPersistableBundle(@NonNull PersistableBundle in) {
Objects.requireNonNull(in, "PersistableBundle is null");
return new EapAkaPrimeConfig(
in.getInt(SUB_ID_KEY),
in.getInt(APP_TYPE_KEY),
in.getString(NETWORK_NAME_KEY),
in.getBoolean(ALL_MISMATCHED_NETWORK_KEY));
}
/**
* Serializes this object to a PersistableBundle
*
* @hide
*/
@Override
@NonNull
protected PersistableBundle toPersistableBundle() {
final PersistableBundle result = super.toPersistableBundle();
result.putString(NETWORK_NAME_KEY, mNetworkName);
result.putBoolean(ALL_MISMATCHED_NETWORK_KEY, mAllowMismatchedNetworkNames);
return result;
}
/**
* Retrieves the network name
*
* return the network name
*/
@NonNull
public String getNetworkName() {
return mNetworkName;
}
/**
* Checks if mismatched network names are allowed
*
* @return whether network name mismatches are allowed
*/
public boolean allowsMismatchedNetworkNames() {
return mAllowMismatchedNetworkNames;
}
/** @hide */
@Override
public int hashCode() {
return Objects.hash(super.hashCode(), mNetworkName, mAllowMismatchedNetworkNames);
}
/** @hide */
@Override
public boolean equals(Object o) {
if (!super.equals(o) || !(o instanceof EapAkaPrimeConfig)) {
return false;
}
EapAkaPrimeConfig other = (EapAkaPrimeConfig) o;
return mNetworkName.equals(other.mNetworkName)
&& mAllowMismatchedNetworkNames == other.mAllowMismatchedNetworkNames;
}
}
/**
* EapMsChapV2Config represents the configs needed for an EAP MSCHAPv2 session.
*/
public static class EapMsChapV2Config extends EapMethodConfig {
private static final String USERNAME_KEY = "username";
private static final String PASSWORD_KEY = "password";
@NonNull private final String mUsername;
@NonNull private final String mPassword;
/** @hide */
@VisibleForTesting
public EapMsChapV2Config(String username, String password) {
super(EAP_TYPE_MSCHAP_V2);
Objects.requireNonNull(username, "username must not be null");
Objects.requireNonNull(password, "password must not be null");
mUsername = username;
mPassword = password;
}
/**
* Constructs this object by deserializing a PersistableBundle
*
* @hide
*/
@NonNull
public static EapMsChapV2Config fromPersistableBundle(@NonNull PersistableBundle in) {
Objects.requireNonNull(in, "PersistableBundle is null");
return new EapMsChapV2Config(in.getString(USERNAME_KEY), in.getString(PASSWORD_KEY));
}
/**
* Serializes this object to a PersistableBundle
*
* @hide
*/
@Override
@NonNull
protected PersistableBundle toPersistableBundle() {
final PersistableBundle result = super.toPersistableBundle();
result.putString(USERNAME_KEY, mUsername);
result.putString(PASSWORD_KEY, mPassword);
return result;
}
/**
* Retrieves the username
*
* @return the username to be used by MSCHAPV2
*/
@NonNull
public String getUsername() {
return mUsername;
}
/**
* Retrieves the password
*
* @return the password to be used by MSCHAPV2
*/
@NonNull
public String getPassword() {
return mPassword;
}
/** @hide */
@Override
public int hashCode() {
return Objects.hash(super.hashCode(), mUsername, mPassword);
}
/** @hide */
@Override
public boolean equals(Object o) {
if (!super.equals(o) || !(o instanceof EapMsChapV2Config)) {
return false;
}
EapMsChapV2Config other = (EapMsChapV2Config) o;
return mUsername.equals(other.mUsername) && mPassword.equals(other.mPassword);
}
}
/**
* EapTtlsConfig represents the configs needed for an EAP-TTLS session.
*/
public static class EapTtlsConfig extends EapMethodConfig {
private static final String TRUST_CERT_KEY = "TRUST_CERT_KEY";
private static final String EAP_SESSION_CONFIG_KEY = "EAP_SESSION_CONFIG_KEY";
@Nullable private final TrustAnchor mOverrideTrustAnchor;
@NonNull private final EapSessionConfig mInnerEapSessionConfig;
/** @hide */
@VisibleForTesting
public EapTtlsConfig(
@Nullable X509Certificate serverCaCert,
@NonNull EapSessionConfig innerEapSessionConfig) {
super(EAP_TYPE_TTLS);
mInnerEapSessionConfig =
Objects.requireNonNull(
innerEapSessionConfig, "innerEapSessionConfig must not be null");
if (mInnerEapSessionConfig.getEapConfigs().containsKey(EAP_TYPE_TTLS)) {
throw new IllegalArgumentException("Recursive EAP-TTLS method configs not allowed");
}
mOverrideTrustAnchor =
(serverCaCert == null)
? null
: new TrustAnchor(serverCaCert, null /* nameConstraints */);
}
/**
* Constructs this object by deserializing a PersistableBundle.
*
* @hide
*/
@NonNull
public static EapTtlsConfig fromPersistableBundle(@NonNull PersistableBundle in) {
Objects.requireNonNull(in, "PersistableBundle is null");
PersistableBundle trustCertBundle = in.getPersistableBundle(TRUST_CERT_KEY);
X509Certificate caCert = null;
if (trustCertBundle != null) {
byte[] encodedCert = PersistableBundleUtils.toByteArray(trustCertBundle);
caCert = IkeCertUtils.certificateFromByteArray(encodedCert);
}
PersistableBundle eapSessionConfigBundle =
in.getPersistableBundle(EAP_SESSION_CONFIG_KEY);
Objects.requireNonNull(eapSessionConfigBundle, "eapSessionConfigBundle is null");
EapSessionConfig eapSessionConfig =
EapSessionConfig.fromPersistableBundle(eapSessionConfigBundle);
return new EapTtlsConfig(caCert, eapSessionConfig);
}
/**
* Serializes this object to a PersistableBundle.
*
* @hide
*/
@Override
@NonNull
protected PersistableBundle toPersistableBundle() {
final PersistableBundle result = super.toPersistableBundle();
try {
if (mOverrideTrustAnchor != null) {
result.putPersistableBundle(
TRUST_CERT_KEY,
PersistableBundleUtils.fromByteArray(
mOverrideTrustAnchor.getTrustedCert().getEncoded()));
}
result.putPersistableBundle(
EAP_SESSION_CONFIG_KEY, mInnerEapSessionConfig.toPersistableBundle());
} catch (CertificateEncodingException e) {
throw new IllegalArgumentException("Fail to encode the certificate");
}
return result;
}
/** @hide */
@Override
public boolean isEapOnlySafeMethod() {
return true;
}
/**
* Retrieves the provided CA certificate for validating the remote certificate(s)
*
* @return the CA certificate for validating the received server certificate or null if the
* system default is preferred
*/
@Nullable
public X509Certificate getServerCaCert() {
return (mOverrideTrustAnchor == null) ? null : mOverrideTrustAnchor.getTrustedCert();
}
/**
* Retrieves the inner EAP session config
*
* @return an EapSessionConfig representing the config for tunneled EAP authentication
*/
@NonNull
public EapSessionConfig getInnerEapSessionConfig() {
return mInnerEapSessionConfig;
}
/** @hide */
@Override
public int hashCode() {
// Use #getTrustedCert() because TrustAnchor does not override #hashCode()
return Objects.hash(
super.hashCode(),
mInnerEapSessionConfig,
(mOverrideTrustAnchor == null) ? null : mOverrideTrustAnchor.getTrustedCert());
}
/** @hide */
@Override
public boolean equals(Object o) {
if (!super.equals(o) || !(o instanceof EapTtlsConfig)) {
return false;
}
EapTtlsConfig other = (EapTtlsConfig) o;
if (!Objects.equals(mInnerEapSessionConfig, other.mInnerEapSessionConfig)) {
return false;
}
if (mOverrideTrustAnchor == null && other.mOverrideTrustAnchor == null) {
return true;
}
return mOverrideTrustAnchor != null
&& other.mOverrideTrustAnchor != null
&& Objects.equals(
mOverrideTrustAnchor.getTrustedCert(),
other.mOverrideTrustAnchor.getTrustedCert());
}
}
/**
* EapAkaOption represents optional configurations for EAP AKA authentication.
*/
public static final class EapAkaOption {
/** @hide */
private static final String REAUTH_ID_KEY = "reauthId";
/** @hide */
private final byte[] mReauthId;
/** @hide */
@VisibleForTesting
public EapAkaOption(@Nullable byte[] reauthId) {
if (reauthId != null) {
mReauthId = new byte[reauthId.length];
System.arraycopy(reauthId, 0, mReauthId, 0, reauthId.length);
} else {
mReauthId = null;
}
}
/**
* Constructs this object by deserializing a PersistableBundle
*
* @hide
*/
@NonNull
public static EapAkaOption fromPersistableBundle(@NonNull PersistableBundle in) {
Objects.requireNonNull(in, "PersistableBundle is null");
EapAkaOption.Builder builder = new EapAkaOption.Builder();
PersistableBundle reauthIdBundle = in.getPersistableBundle(REAUTH_ID_KEY);
if (reauthIdBundle != null) {
byte[] reauthId = PersistableBundleUtils.toByteArray(reauthIdBundle);
builder.setReauthId(reauthId);
}
return builder.build();
}
/**
* Serializes this object to a PersistableBundle
*
* @hide
*/
@NonNull
protected PersistableBundle toPersistableBundle() {
final PersistableBundle result = new PersistableBundle();
if (mReauthId != null) {
result.putPersistableBundle(
REAUTH_ID_KEY, PersistableBundleUtils.fromByteArray(mReauthId));
}
return result;
}
/**
* Retrieves the re-authentication ID
*
* @return the re-authentication ID
*/
@Nullable
public byte[] getReauthId() {
return mReauthId;
}
/** @hide */
@Override
public int hashCode() {
return Objects.hash(super.hashCode(), Arrays.hashCode(mReauthId));
}
/** @hide */
@Override
public boolean equals(Object o) {
if (!(o instanceof EapAkaOption)) {
return false;
}
EapAkaOption other = (EapAkaOption) o;
return Arrays.equals(mReauthId, other.mReauthId);
}
/**
* This class can be used to incrementally construct an {@link EapAkaOption}.
*/
public static final class Builder {
byte[] mReauthId;
/**
* Set fast re-authentication ID
*
* If keys are found matching the combination of reauthId and permanent ID,
* re-authentication will be attempted.
*
* Permanent ID MUST be set in setEapIdentity
*
* Upon session establishment, new re-authentication IDs will be listed in the
* EapAkaInfo returned as part of IkeSessionCallback#onOpened().
*
* Reauthentication is generally considered less secure, as it does not prove the
* existence of the full credentials, and should be used only when a strong correlation
* can be provided to the full authentication (eg shared keys from previous
* authentication runs)
*
* @see RFC 4186,
* Extensible Authentication Protocol Method for 3rd Generation Authentication and
* Key Agreement (EAP-AKA)
*
* @param reauthId re-authentication ID encoded with UTF-8
* @return Builder this, to facilitate chaining.
*/
@NonNull
public Builder setReauthId(@NonNull byte[] reauthId) {
mReauthId = reauthId;
return this;
}
/**
* Constructs and returns an EapAkaOption with the configurations applied to this
* Builder.
*
* @return the EapAkaOption constructed by this Builder.
*/
@NonNull
public EapAkaOption build() {
return new EapAkaOption(mReauthId);
}
}
}
/**
* Checks if all the methods in the session are EAP-only safe
*
* @return whether all the methods in the session are EAP-only safe
*
* @see RFC 5998#section 4, for safe eap
* methods
*
* @hide
*/
public boolean areAllMethodsEapOnlySafe() {
for (Map.Entry