// Copyright 2016 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package org.chromium.device.sensors;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertNotNull;
import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertTrue;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.ArgumentMatchers.anyInt;
import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.doReturn;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.spy;
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import android.content.Context;
import android.hardware.Sensor;
import android.hardware.SensorEvent;
import android.hardware.SensorEventListener;
import android.hardware.SensorManager;
import android.os.Handler;
import android.util.SparseArray;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.MockitoAnnotations;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.stubbing.Answer;
import org.robolectric.annotation.Config;
import org.chromium.base.FeatureList;
import org.chromium.base.test.BaseRobolectricTestRunner;
import org.chromium.base.test.util.Feature;
import org.chromium.device.mojom.ReportingMode;
import org.chromium.device.mojom.SensorType;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.util.ArrayList;
import java.util.List;
/** Unit tests for PlatformSensor and PlatformSensorProvider. */
@RunWith(BaseRobolectricTestRunner.class)
@Config(manifest = Config.NONE)
@SuppressWarnings("GuardedBy") // verify(sensor, times(1)).sensorError() cannot resolve |mLock|.
public class PlatformSensorAndProviderTest {
@Mock private Context mContext;
@Mock private SensorManager mSensorManager;
@Mock private PlatformSensorProvider mPlatformSensorProvider;
private final SparseArray<List<Sensor>> mMockSensors = new SparseArray<>();
private static final long PLATFORM_SENSOR_ANDROID = 123456789L;
private static final long PLATFORM_SENSOR_TIMESTAMP = 314159265358979L;
private static final double SECONDS_IN_NANOSECOND = 0.000000001d;
@SuppressWarnings("LockNotBeforeTry")
/** Class that overrides thread management callbacks for testing purposes. */
private static class TestPlatformSensorProvider extends PlatformSensorProvider {
public TestPlatformSensorProvider(Context context) {
super(context);
}
@Override
public Handler getHandler() {
return new Handler();
}
@Override
protected void startSensorThread() {}
@Override
protected void stopSensorThread() {}
}
/** Class that overrides native callbacks for testing purposes. */
private static class TestPlatformSensor extends PlatformSensor {
public TestPlatformSensor(
Sensor sensor, int readingCount, PlatformSensorProvider provider) {
super(sensor, readingCount, provider, PLATFORM_SENSOR_ANDROID);
}
@Override
protected void updateSensorReading(
double timestamp, double value1, double value2, double value3, double value4) {}
@Override
protected void sensorError() {}
}
@Before
public void setUp() {
FeatureList.TestValues testValues = new FeatureList.TestValues();
FeatureList.setTestValues(testValues);
MockitoAnnotations.initMocks(this);
// Remove all mock sensors before the test.
mMockSensors.clear();
doReturn(mSensorManager).when(mContext).getSystemService(Context.SENSOR_SERVICE);
doAnswer(
new Answer<List<Sensor>>() {
@Override
public List<Sensor> answer(final InvocationOnMock invocation) {
return getMockSensors(
(int) (Integer) (invocation.getArguments())[0]);
}
})
.when(mSensorManager)
.getSensorList(anyInt());
doReturn(mSensorManager).when(mPlatformSensorProvider).getSensorManager();
doReturn(new Handler()).when(mPlatformSensorProvider).getHandler();
// By default, allow successful registration of SensorEventListeners.
doReturn(true)
.when(mSensorManager)
.registerListener(
any(SensorEventListener.class),
any(Sensor.class),
anyInt(),
any(Handler.class));
}
/** Test that PlatformSensorProvider cannot create sensors if sensor manager is null. */
@Test
@Feature({"PlatformSensorProvider"})
public void testNullSensorManager() {
doReturn(null).when(mContext).getSystemService(Context.SENSOR_SERVICE);
PlatformSensorProvider provider = PlatformSensorProvider.createForTest(mContext);
PlatformSensor sensor =
PlatformSensor.create(provider, SensorType.AMBIENT_LIGHT, PLATFORM_SENSOR_ANDROID);
assertNull(sensor);
}
/** Test that PlatformSensorProvider cannot create sensors that are not supported. */
@Test
@Feature({"PlatformSensorProvider"})
public void testSensorNotSupported() {
PlatformSensorProvider provider = PlatformSensorProvider.createForTest(mContext);
PlatformSensor sensor =
PlatformSensor.create(provider, SensorType.AMBIENT_LIGHT, PLATFORM_SENSOR_ANDROID);
assertNull(sensor);
}
/**
* Test that PlatformSensorProvider maps device::SensorType to android.hardware.Sensor.TYPE_*.
*/
@Test
@Feature({"PlatformSensorProvider"})
public void testSensorTypeMappings() {
PlatformSensorProvider provider = PlatformSensorProvider.createForTest(mContext);
PlatformSensor.create(provider, SensorType.AMBIENT_LIGHT, PLATFORM_SENSOR_ANDROID);
verify(mSensorManager).getSensorList(Sensor.TYPE_LIGHT);
PlatformSensor.create(provider, SensorType.ACCELEROMETER, PLATFORM_SENSOR_ANDROID);
verify(mSensorManager).getSensorList(Sensor.TYPE_ACCELEROMETER);
PlatformSensor.create(provider, SensorType.LINEAR_ACCELERATION, PLATFORM_SENSOR_ANDROID);
verify(mSensorManager).getSensorList(Sensor.TYPE_LINEAR_ACCELERATION);
PlatformSensor.create(provider, SensorType.GRAVITY, PLATFORM_SENSOR_ANDROID);
verify(mSensorManager).getSensorList(Sensor.TYPE_GRAVITY);
PlatformSensor.create(provider, SensorType.GYROSCOPE, PLATFORM_SENSOR_ANDROID);
verify(mSensorManager).getSensorList(Sensor.TYPE_GYROSCOPE);
PlatformSensor.create(provider, SensorType.MAGNETOMETER, PLATFORM_SENSOR_ANDROID);
verify(mSensorManager).getSensorList(Sensor.TYPE_MAGNETIC_FIELD);
PlatformSensor.create(
provider, SensorType.ABSOLUTE_ORIENTATION_QUATERNION, PLATFORM_SENSOR_ANDROID);
verify(mSensorManager).getSensorList(Sensor.TYPE_ROTATION_VECTOR);
PlatformSensor.create(
provider, SensorType.RELATIVE_ORIENTATION_QUATERNION, PLATFORM_SENSOR_ANDROID);
verify(mSensorManager).getSensorList(Sensor.TYPE_GAME_ROTATION_VECTOR);
}
/** Test that PlatformSensorProvider can create sensors that are supported. */
@Test
@Feature({"PlatformSensorProvider"})
public void testSensorSupported() {
PlatformSensor sensor =
createPlatformSensor(
50000,
Sensor.TYPE_LIGHT,
SensorType.AMBIENT_LIGHT,
Sensor.REPORTING_MODE_ON_CHANGE);
assertNotNull(sensor);
sensor.sensorDestroyed();
}
/**
* Test that PlatformSensor notifies PlatformSensorProvider when it starts (stops) polling,
* and SensorEventListener is registered (unregistered) to sensor manager.
*/
@Test
@Feature({"PlatformSensor"})
public void testSensorStartStop() {
addMockSensor(50000, Sensor.TYPE_ACCELEROMETER, Sensor.REPORTING_MODE_CONTINUOUS);
PlatformSensor sensor =
PlatformSensor.create(
mPlatformSensorProvider, SensorType.ACCELEROMETER, PLATFORM_SENSOR_ANDROID);
assertNotNull(sensor);
sensor.startSensor(5);
sensor.stopSensor();
// Multiple start invocations.
sensor.startSensor(1);
sensor.startSensor(2);
sensor.startSensor(3);
// Same frequency, should not restart sensor
sensor.startSensor(3);
// Started polling with 5, 1, 2 and 3 Hz frequency.
verify(mPlatformSensorProvider, times(4)).getHandler();
verify(mPlatformSensorProvider, times(4)).sensorStarted(sensor);
verify(mSensorManager, times(4))
.registerListener(
any(SensorEventListener.class),
any(Sensor.class),
anyInt(),
any(Handler.class));
sensor.stopSensor();
sensor.stopSensor();
verify(mPlatformSensorProvider, times(3)).sensorStopped(sensor);
verify(mSensorManager, times(4))
.unregisterListener(any(SensorEventListener.class), any(Sensor.class));
sensor.sensorDestroyed();
}
/**
* Test that PlatformSensorProvider is notified when PlatformSensor starts and in case of
* failure, tells PlatformSensorProvider that the sensor is stopped, so that polling thread
* can be stopped.
*/
@Test
@Feature({"PlatformSensor"})
public void testSensorStartFails() {
TestPlatformSensor sensor =
createTestPlatformSensor(
50000, Sensor.TYPE_ACCELEROMETER, 3, Sensor.REPORTING_MODE_CONTINUOUS);
TestPlatformSensor spySensor = spy(sensor);
// Accelerometer requires 3 reading values x,y and z, create fake event with 1 reading
// value.
SensorEvent event = createFakeEvent(1);
assertNotNull(event);
spySensor.onSensorChanged(event);
doReturn(false)
.when(mSensorManager)
.registerListener(
any(SensorEventListener.class),
any(Sensor.class),
anyInt(),
any(Handler.class));
spySensor.startSensor(5);
verify(mPlatformSensorProvider, times(1)).sensorStarted(spySensor);
verify(mPlatformSensorProvider, times(2)).sensorStopped(spySensor);
verify(mPlatformSensorProvider, times(1)).getHandler();
verify(spySensor, times(2)).sensorError();
sensor.sensorDestroyed();
}
/** Same as the above except instead of a clean failure an exception is thrown. */
@Test
@Feature({"PlatformSensor"})
public void testSensorStartFailsWithException() {
TestPlatformSensor sensor =
createTestPlatformSensor(
50000, Sensor.TYPE_ACCELEROMETER, 3, Sensor.REPORTING_MODE_CONTINUOUS);
TestPlatformSensor spySensor = spy(sensor);
// Accelerometer requires 3 reading values x,y and z, create fake event with 1 reading
// value.
SensorEvent event = createFakeEvent(1);
assertNotNull(event);
spySensor.onSensorChanged(event);
when(mSensorManager.registerListener(
any(SensorEventListener.class),
any(Sensor.class),
anyInt(),
any(Handler.class)))
.thenThrow(RuntimeException.class);
spySensor.startSensor(5);
verify(mPlatformSensorProvider, times(1)).sensorStarted(spySensor);
verify(mPlatformSensorProvider, times(2)).sensorStopped(spySensor);
verify(mPlatformSensorProvider, times(1)).getHandler();
verify(spySensor, times(2)).sensorError();
sensor.sensorDestroyed();
}
/** Test that PlatformSensor correctly checks supported configuration. */
@Test
@Feature({"PlatformSensor"})
public void testSensorConfiguration() {
// 5Hz min delay
PlatformSensor sensor =
createPlatformSensor(
200000,
Sensor.TYPE_ACCELEROMETER,
SensorType.ACCELEROMETER,
Sensor.REPORTING_MODE_CONTINUOUS);
assertTrue(sensor.checkSensorConfiguration(5));
assertFalse(sensor.checkSensorConfiguration(6));
sensor.sensorDestroyed();
}
/** Test that PlatformSensor correctly returns its reporting mode. */
@Test
@Feature({"PlatformSensor"})
public void testSensorOnChangeReportingMode() {
PlatformSensor sensor =
createPlatformSensor(
50000,
Sensor.TYPE_LIGHT,
SensorType.AMBIENT_LIGHT,
Sensor.REPORTING_MODE_ON_CHANGE);
assertEquals(ReportingMode.ON_CHANGE, sensor.getReportingMode());
sensor.sensorDestroyed();
}
/** Test that PlatformSensor correctly returns its maximum supported frequency. */
@Test
@Feature({"PlatformSensor"})
public void testSensorMaximumSupportedFrequency() {
PlatformSensor sensor =
createPlatformSensor(
50000,
Sensor.TYPE_LIGHT,
SensorType.AMBIENT_LIGHT,
Sensor.REPORTING_MODE_ON_CHANGE);
assertEquals(20, sensor.getMaximumSupportedFrequency(), 0.001);
sensor.sensorDestroyed();
sensor =
createPlatformSensor(
0,
Sensor.TYPE_LIGHT,
SensorType.AMBIENT_LIGHT,
Sensor.REPORTING_MODE_ON_CHANGE);
assertEquals(
sensor.getDefaultConfiguration(), sensor.getMaximumSupportedFrequency(), 0.001);
sensor.sensorDestroyed();
}
/** Test that shared buffer is correctly populated from SensorEvent. */
@Test
@Feature({"PlatformSensor"})
public void testSensorReadingFromEvent() {
TestPlatformSensor sensor =
createTestPlatformSensor(
50000, Sensor.TYPE_LIGHT, 1, Sensor.REPORTING_MODE_ON_CHANGE);
TestPlatformSensor spySensor = spy(sensor);
SensorEvent event = createFakeEvent(1);
assertNotNull(event);
spySensor.onSensorChanged(event);
double timestamp = PLATFORM_SENSOR_TIMESTAMP * SECONDS_IN_NANOSECOND;
verify(spySensor, times(1))
.updateSensorReading(timestamp, getFakeReadingValue(1), 0.0, 0.0, 0.0);
sensor.sensorDestroyed();
}
/**
* Test that shared buffer is correctly populated from SensorEvent for sensors with more
* than one value.
*/
@Test
@Feature({"PlatformSensor"})
public void testSensorReadingFromEventMoreValues() {
TestPlatformSensor sensor =
createTestPlatformSensor(
50000, Sensor.TYPE_ROTATION_VECTOR, 4, Sensor.REPORTING_MODE_ON_CHANGE);
TestPlatformSensor spySensor = spy(sensor);
SensorEvent event = createFakeEvent(4);
assertNotNull(event);
spySensor.onSensorChanged(event);
double timestamp = PLATFORM_SENSOR_TIMESTAMP * SECONDS_IN_NANOSECOND;
verify(spySensor, times(1))
.updateSensorReading(
timestamp,
getFakeReadingValue(1),
getFakeReadingValue(2),
getFakeReadingValue(3),
getFakeReadingValue(4));
sensor.sensorDestroyed();
}
/** Test that PlatformSensor notifies client when there is an error. */
@Test
@Feature({"PlatformSensor"})
public void testSensorInvalidReadingSize() {
TestPlatformSensor sensor =
createTestPlatformSensor(
50000, Sensor.TYPE_ACCELEROMETER, 3, Sensor.REPORTING_MODE_CONTINUOUS);
TestPlatformSensor spySensor = spy(sensor);
// Accelerometer requires 3 reading values x,y and z, create fake event with 1 reading
// value.
SensorEvent event = createFakeEvent(1);
assertNotNull(event);
spySensor.onSensorChanged(event);
verify(spySensor, times(1)).sensorError();
sensor.sensorDestroyed();
}
/**
* Test that multiple PlatformSensor instances correctly register (unregister) to
* sensor manager and notify PlatformSensorProvider when they start (stop) polling for data.
*/
@Test
@Feature({"PlatformSensor"})
public void testMultipleSensorTypeInstances() {
addMockSensor(200000, Sensor.TYPE_LIGHT, Sensor.REPORTING_MODE_ON_CHANGE);
addMockSensor(50000, Sensor.TYPE_ACCELEROMETER, Sensor.REPORTING_MODE_CONTINUOUS);
TestPlatformSensorProvider spyProvider = spy(new TestPlatformSensorProvider(mContext));
PlatformSensor lightSensor =
PlatformSensor.create(
spyProvider, SensorType.AMBIENT_LIGHT, PLATFORM_SENSOR_ANDROID);
assertNotNull(lightSensor);
PlatformSensor accelerometerSensor =
PlatformSensor.create(
spyProvider, SensorType.ACCELEROMETER, PLATFORM_SENSOR_ANDROID);
assertNotNull(accelerometerSensor);
lightSensor.startSensor(3);
accelerometerSensor.startSensor(10);
lightSensor.stopSensor();
accelerometerSensor.stopSensor();
verify(spyProvider, times(2)).getHandler();
verify(spyProvider, times(1)).sensorStarted(lightSensor);
verify(spyProvider, times(1)).sensorStarted(accelerometerSensor);
verify(spyProvider, times(1)).sensorStopped(lightSensor);
verify(spyProvider, times(1)).sensorStopped(accelerometerSensor);
verify(spyProvider, times(1)).startSensorThread();
verify(spyProvider, times(1)).stopSensorThread();
verify(mSensorManager, times(2))
.registerListener(
any(SensorEventListener.class),
any(Sensor.class),
anyInt(),
any(Handler.class));
verify(mSensorManager, times(2))
.unregisterListener(any(SensorEventListener.class), any(Sensor.class));
lightSensor.sensorDestroyed();
accelerometerSensor.sensorDestroyed();
}
/**
* Creates fake event. The SensorEvent constructor is not accessible outside android.hardware
* package, therefore, java reflection is used to make constructor accessible to construct
* SensorEvent instance.
*/
private SensorEvent createFakeEvent(int readingValuesNum) {
try {
Constructor<SensorEvent> sensorEventConstructor =
SensorEvent.class.getDeclaredConstructor(Integer.TYPE);
sensorEventConstructor.setAccessible(true);
SensorEvent event = sensorEventConstructor.newInstance(readingValuesNum);
event.timestamp = PLATFORM_SENSOR_TIMESTAMP;
for (int i = 0; i < readingValuesNum; ++i) {
event.values[i] = getFakeReadingValue(i + 1);
}
return event;
} catch (InvocationTargetException
| NoSuchMethodException
| InstantiationException
| IllegalAccessException e) {
return null;
}
}
private void addMockSensor(long minDelayUsec, int sensorType, int reportingMode) {
List<Sensor> mockSensorList = new ArrayList<Sensor>();
mockSensorList.add(createMockSensor(minDelayUsec, sensorType, reportingMode));
mMockSensors.put(sensorType, mockSensorList);
}
private Sensor createMockSensor(long minDelayUsec, int sensorType, int reportingMode) {
Sensor mockSensor = mock(Sensor.class);
doReturn((int) minDelayUsec).when(mockSensor).getMinDelay();
doReturn(reportingMode).when(mockSensor).getReportingMode();
doReturn(sensorType).when(mockSensor).getType();
return mockSensor;
}
private List<Sensor> getMockSensors(int sensorType) {
if (mMockSensors.indexOfKey(sensorType) >= 0) {
return mMockSensors.get(sensorType);
}
return new ArrayList<Sensor>();
}
private PlatformSensor createPlatformSensor(
long minDelayUsec, int androidSensorType, int mojoSensorType, int reportingMode) {
addMockSensor(minDelayUsec, androidSensorType, reportingMode);
PlatformSensorProvider provider = PlatformSensorProvider.createForTest(mContext);
return PlatformSensor.create(provider, mojoSensorType, PLATFORM_SENSOR_ANDROID);
}
private TestPlatformSensor createTestPlatformSensor(
long minDelayUsec, int androidSensorType, int readingCount, int reportingMode) {
return new TestPlatformSensor(
createMockSensor(minDelayUsec, androidSensorType, reportingMode),
readingCount,
mPlatformSensorProvider);
}
private float getFakeReadingValue(int valueNum) {
return (float) (valueNum + SECONDS_IN_NANOSECOND);
}
}