blob: af6848f29bc97b9e0c864766971568ce01ef5d28 [file] [log] [blame]
/*
* Copyright (C) 2020 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 com.android.internal.net.eap.statemachine;
import static android.net.eap.EapSessionConfig.EapMethodConfig.EAP_TYPE_TTLS;
import static androidx.test.InstrumentationRegistry.getInstrumentation;
import static com.android.internal.net.TestUtils.hexStringToByteArray;
import static com.android.internal.net.eap.crypto.TlsSession.TLS_STATUS_CLOSED;
import static com.android.internal.net.eap.message.EapData.EAP_NOTIFICATION;
import static com.android.internal.net.eap.message.EapMessage.EAP_CODE_FAILURE;
import static com.android.internal.net.eap.message.EapMessage.EAP_CODE_REQUEST;
import static com.android.internal.net.eap.message.EapMessage.EAP_CODE_SUCCESS;
import static com.android.internal.net.eap.message.EapTestMessageDefinitions.EAP_RESPONSE_NOTIFICATION_PACKET;
import static com.android.internal.net.eap.message.EapTestMessageDefinitions.EAP_RESPONSE_TTLS_WITH_LENGTH;
import static com.android.internal.net.eap.message.EapTestMessageDefinitions.EAP_TTLS_DUMMY_DATA_BYTES;
import static com.android.internal.net.eap.message.EapTestMessageDefinitions.EAP_TTLS_DUMMY_DATA_FINAL_FRAGMENT_BYTES;
import static com.android.internal.net.eap.message.EapTestMessageDefinitions.EAP_TTLS_DUMMY_DATA_INITIAL_FRAGMENT_BYTES;
import static com.android.internal.net.eap.message.EapTestMessageDefinitions.ID_INT;
import static com.android.internal.net.eap.message.EapTestMessageDefinitions.MSCHAP_V2_PASSWORD;
import static com.android.internal.net.eap.message.EapTestMessageDefinitions.MSCHAP_V2_USERNAME;
import static org.junit.Assert.assertArrayEquals;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.eq;
import static org.mockito.Mockito.any;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import android.content.Context;
import android.net.eap.EapSessionConfig;
import android.net.eap.EapSessionConfig.EapTtlsConfig;
import com.android.internal.net.eap.EapResult;
import com.android.internal.net.eap.EapResult.EapError;
import com.android.internal.net.eap.EapResult.EapFailure;
import com.android.internal.net.eap.EapResult.EapResponse;
import com.android.internal.net.eap.crypto.TlsSession;
import com.android.internal.net.eap.crypto.TlsSession.TlsResult;
import com.android.internal.net.eap.crypto.TlsSessionFactory;
import com.android.internal.net.eap.exceptions.EapInvalidRequestException;
import com.android.internal.net.eap.message.EapData;
import com.android.internal.net.eap.message.EapMessage;
import com.android.internal.net.eap.message.ttls.EapTtlsInboundFragmentationHelper;
import com.android.internal.net.eap.message.ttls.EapTtlsOutboundFragmentationHelper;
import com.android.internal.net.eap.message.ttls.EapTtlsTypeData;
import com.android.internal.net.eap.message.ttls.EapTtlsTypeData.EapTtlsTypeDataDecoder;
import com.android.internal.net.eap.message.ttls.EapTtlsTypeData.EapTtlsTypeDataDecoder.DecodeResult;
import com.android.internal.net.eap.statemachine.EapMethodStateMachine.EapMethodState;
import com.android.internal.net.eap.statemachine.EapMethodStateMachine.FinalState;
import com.android.internal.net.eap.statemachine.EapTtlsMethodStateMachine.ErroredAndAwaitingClosureState;
import org.junit.AfterClass;
import org.junit.Before;
import org.junit.Test;
import java.security.SecureRandom;
public class EapTtlsStateTest {
static final String NOTIFICATION_MESSAGE = "test";
static final byte[] DUMMY_EAP_TYPE_DATA = hexStringToByteArray("112233445566");
static final byte[] EMPTY_BYTE_ARRAY = new byte[0];
static final int BUFFER_SIZE_FRAGMENT_ONE = EAP_TTLS_DUMMY_DATA_INITIAL_FRAGMENT_BYTES.length;
static final int BUFFER_SIZE_FRAGMENT_TWO = EAP_TTLS_DUMMY_DATA_FINAL_FRAGMENT_BYTES.length;
static final int BUFFER_SIZE_ASSEMBLED_FRAGMENTS =
BUFFER_SIZE_FRAGMENT_ONE + BUFFER_SIZE_FRAGMENT_TWO;
Context mContext;
SecureRandom mMockSecureRandom;
EapTtlsTypeDataDecoder mMockTypeDataDecoder;
TlsSessionFactory mMockTlsSessionFactory;
TlsSession mMockTlsSession;
EapTtlsConfig mEapTtlsConfig;
EapTtlsMethodStateMachine mStateMachine;
EapTtlsInboundFragmentationHelper mInboundFragmentationHelper;
EapTtlsOutboundFragmentationHelper mOutboundFragmentationHelper;
@Before
public void setUp() throws Exception {
mContext = getInstrumentation().getContext();
mMockSecureRandom = mock(SecureRandom.class);
mMockTypeDataDecoder = mock(EapTtlsTypeDataDecoder.class);
mMockTlsSessionFactory = mock(TlsSessionFactory.class);
mMockTlsSession = mock(TlsSession.class);
EapSessionConfig innerEapSessionConfig =
new EapSessionConfig.Builder()
.setEapMsChapV2Config(MSCHAP_V2_USERNAME, MSCHAP_V2_PASSWORD)
.build();
EapSessionConfig eapSessionConfig =
new EapSessionConfig.Builder()
.setEapTtlsConfig(null /* trustedCa */, innerEapSessionConfig)
.build();
mEapTtlsConfig = eapSessionConfig.getEapTtlsConfig();
mInboundFragmentationHelper = new EapTtlsInboundFragmentationHelper();
mOutboundFragmentationHelper =
new EapTtlsOutboundFragmentationHelper(BUFFER_SIZE_FRAGMENT_ONE);
mStateMachine =
new EapTtlsMethodStateMachine(
mContext,
mEapTtlsConfig,
mMockSecureRandom,
mMockTypeDataDecoder,
mInboundFragmentationHelper,
mOutboundFragmentationHelper);
EapTtlsMethodStateMachine.sTlsSessionFactory = mMockTlsSessionFactory;
when(mMockTlsSessionFactory.newInstance(any(), any())).thenReturn(mMockTlsSession);
}
@AfterClass
public static void teardown() {
EapTtlsMethodStateMachine.sTlsSessionFactory = new TlsSessionFactory();
}
@Test
public void testHandleEapFailureNotification() throws Exception {
EapResult result = mStateMachine.process(new EapMessage(EAP_CODE_FAILURE, ID_INT, null));
assertTrue(result instanceof EapFailure);
assertTrue(mStateMachine.getState() instanceof FinalState);
}
@Test
public void testHandleEapSuccessNotification() throws Exception {
EapResult result = mStateMachine.process(new EapMessage(EAP_CODE_SUCCESS, ID_INT, null));
EapError eapError = (EapError) result;
assertTrue(eapError.cause instanceof EapInvalidRequestException);
}
@Test
public void testHandleEapNotification() throws Exception {
EapData eapData = new EapData(EAP_NOTIFICATION, NOTIFICATION_MESSAGE.getBytes());
EapMessage eapMessage = new EapMessage(EAP_CODE_REQUEST, ID_INT, eapData);
EapMethodState preNotification = (EapMethodState) mStateMachine.getState();
EapResult result = mStateMachine.process(eapMessage);
assertEquals(preNotification, mStateMachine.getState());
EapResponse eapResponse = (EapResponse) result;
assertArrayEquals(EAP_RESPONSE_NOTIFICATION_PACKET, eapResponse.packet);
}
EapTtlsTypeData getEapTtlsStartTypeData() throws Exception {
return getEapTtlsTypeData(
false /* isFragmented */, true /* isStart */, 0 /* length */, EMPTY_BYTE_ARRAY);
}
EapTtlsTypeData getEapTtlsFragmentTypeData(boolean isFragment, int length, byte[] data)
throws Exception {
return getEapTtlsTypeData(isFragment, false /* isStart */, length, data);
}
EapTtlsTypeData getEapTtlsTypeData(byte[] data) throws Exception {
return getEapTtlsTypeData(
false /* isFragmented */, false /* isStart */, 0 /* length */, data);
}
EapTtlsTypeData getEapTtlsTypeData(
boolean isFragmented, boolean isStart, int length, byte[] data) throws Exception {
return EapTtlsTypeData.getEapTtlsTypeData(
isFragmented, isStart, 0 /* version */, length, data);
}
void mockTypeDataDecoding(EapTtlsTypeData decodedTypeData) throws Exception {
when(mMockTypeDataDecoder.decodeEapTtlsRequestPacket(eq(DUMMY_EAP_TYPE_DATA)))
.thenReturn(new DecodeResult(decodedTypeData));
}
/** Runs a test and verifies the EAP response returned by the state */
void processMessageAndVerifyEapResponse(byte[] expectedResponse) throws Exception {
EapData eapData = new EapData(EAP_TYPE_TTLS, DUMMY_EAP_TYPE_DATA);
EapMessage eapMessage = new EapMessage(EAP_CODE_REQUEST, ID_INT, eapData);
EapResult result = mStateMachine.process(eapMessage);
EapResponse eapResponse = (EapResponse) result;
assertArrayEquals(expectedResponse, eapResponse.packet);
}
/**
* Completes a run of operations that requires CloseConnection to be called
*
* @param decodedTypeData the type data that is decoded by the type data decoder
*/
void processMessageAndVerifyConnectionClosed(EapTtlsTypeData decodedTypeData) throws Exception {
mockTypeDataDecoding(decodedTypeData);
when(mMockTlsSession.closeConnection())
.thenReturn(
mMockTlsSession
.new TlsResult(EAP_TTLS_DUMMY_DATA_BYTES, TLS_STATUS_CLOSED));
processMessageAndVerifyEapResponse(EAP_RESPONSE_TTLS_WITH_LENGTH);
verify(mMockTypeDataDecoder).decodeEapTtlsRequestPacket(eq(DUMMY_EAP_TYPE_DATA));
verify(mMockTlsSession).closeConnection();
assertTrue(mStateMachine.getState() instanceof ErroredAndAwaitingClosureState);
}
/**
* Runs a test and verifies the EAP error returned by the state
*
* @param error the exception within the EapError
*/
void processMessageAndVerifyEapError(Class<? extends Exception> error) throws Exception {
EapData eapData = new EapData(EAP_TYPE_TTLS, DUMMY_EAP_TYPE_DATA);
EapMessage eapMessage = new EapMessage(EAP_CODE_REQUEST, ID_INT, eapData);
EapResult result = mStateMachine.process(eapMessage);
EapError eapError = (EapError) result;
assertTrue(error.isInstance(eapError.cause));
}
}