Как модульный тестовый код, который использует ZoneId.systemDefault без повышения org.threeten.bp.zone.ZoneRulesException

У меня есть следующий код приложения.

Код приложения

public static long advanceByMax1HourWithoutOverflow(long currentTimeMillis) {
    ZoneId zoneId = ZoneId.systemDefault();

    ZonedDateTime zonedDateTime = Instant.ofEpochMilli(currentTimeMillis).atZone(zoneId);

    // 0-23
    int hour = zonedDateTime.getHour();

    if (hour < 23) {
        zonedDateTime = zonedDateTime.plusHours(1);
        return zonedDateTime.toInstant().toEpochMilli();
    }

    // 0-59
    int minute = zonedDateTime.getMinute();

    if (minute < 59) {
        zonedDateTime = zonedDateTime.plusMinutes(1);
        return zonedDateTime.toInstant().toEpochMilli();
    }

    return currentTimeMillis;
}

Я написал для него единичный тест со следующим.

Единичный тестовый код

@Test
public void advanceByMax1HourWithoutOverflow() {
    /*
    27 October 2018
    1540648800000 -> 1540652400000
    22:00 -> 23:00
    */
    long currentTimeMillis = 1540648800000L;
    long expected = 1540652400000L;
    long output = Utils.advanceByMax1HourWithoutOverflow(currentTimeMillis);
    assertEquals(expected, output);

Я получил следующую ошибку

org.threeten.bp.zone.ZoneRulesException: No time-zone data files registered

Я попытался исправить это, используя

Единичный тестовый код

@Test
public void advanceByMax1HourWithoutOverflow() {
    Context context = mock(Context.class);
    AndroidThreeTen.init(context);

    /*
    27 October 2018
    1540648800000 -> 1540652400000
    22:00 -> 23:00
    */
    long currentTimeMillis = 1540648800000L;
    long expected = 1540652400000L;
    long output = Utils.advanceByMax1HourWithoutOverflow(currentTimeMillis);
    assertEquals(expected, output);

я собираюсь

java.lang.ExceptionInInitializerError
    at org.threeten.bp.ZoneRegion.ofId(ZoneRegion.java:143)
    at org.threeten.bp.ZoneId.of(ZoneId.java:358)
    at org.threeten.bp.ZoneId.of(ZoneId.java:286)
    at org.threeten.bp.ZoneId.systemDefault(ZoneId.java:245)

Есть ли способ передать модульный тест, не меняя подписи моего текущего приложения? Я хотел бы просто сохранить мою текущую подпись метода приложения без каких-либо изменений.

long advanceByMax1HourWithoutOverflow(long currentTimeMillis)

Обратите внимание: градиент, который я использую для модульного тестирования,

implementation 'com.jakewharton.threetenabp:threetenabp:1.1.1'

testImplementation 'junit:junit:4.12'
testImplementation 'org.mockito:mockito-core:2.19.0'

Всего 1 ответ


ThreeTen-Android Backport предоставляет backport для классов Java-времени Java SE 8 для системы Android, которая не поддерживает Java 8. Причина этой проблемы заключается в том, что модульный тест работает на JVM, а не на Android в соответствии с разработчиком Android :

Единичные тесты, которые выполняются только на вашей локальной машине. Эти тесты скомпилированы для локального запуска на виртуальной машине Java (JVM) для минимизации времени выполнения. Если ваши тесты зависят от объектов в платформе Android, мы рекомендуем использовать Robolectric. Для тестов, которые зависят от ваших собственных зависимостей, используйте макеты для эмуляции поведения ваших зависимостей.

Джейк Уортон (владелец твиттер-Android Backport) о том, что не создал ресурсы Robolectric , сказал :

Это ошибка Robolectric. Либо сообщите о них, либо переключитесь на использование обычного TritenBP в своих модульных тестах. Или оба!

В результате мы должны решить эту проблему, используя оригинальную библиотеку ThreeTen Backport. Поэтому мы должны добавить свою зависимость к файлу build.gradle который будет использоваться модульным тестом.

В файле build.gradle:

dependencies {

    implementation 'com.jakewharton.threetenabp:threetenabp:1.1.1'

    testImplementation ('org.threeten:threetenbp:1.3.2'){
        exclude group:'com.jakewharton.threetenabp', module:'threetenabp'
    }
}

В модульном тестовом классе:

@Test
public void advanceByMax1HourWithoutOverflow() {
    /*
    27 October 2018
    1540648800000 -> 1540652400000
    22:00 -> 23:00
    */
    long currentTimeMillis = 1540648800000L;
    long expected = 1540652400000L;
    long output = Utils.advanceByMax1HourWithoutOverflow(currentTimeMillis);
    assertEquals(expected, output);
}

Результат:

введите описание изображения здесь


Есть идеи?

10000