如何模拟在另一个类中使用另一个静态final变量的变量

时间:2018-02-13 15:59:10

标签: java unit-testing junit mocking powermockito

我在junit中测试此方法时遇到问题。我正在尝试使用新的时间戳测试updateConfigDates的方法。请在这里忍受,因为提供了大量信息,但它看起来并不太漂亮。

以下是测试中的方法。

public static Document updateConfigDates(Document doc, Timestamp configDate)
{
    //Format timestamp to string
    String configTimestampStr = GTR_DATE_FORMAT.format(configDate) + "Z";

    //Change configuration date for all nodes in GTR
    NodeList configIDNodes = doc.getElementsByTagName("ConfigDate");
    for (Element cidNode : new DOMUtil.ElementList(configIDNodes))
    {
        cidNode.setTextContent(configTimestampStr);
    }

    return doc;
}

问题在于:

    //Format timestamp to string
    String configTimestampStr = GTR_DATE_FORMAT.format(configDate) + "Z";

哪个使用公共最终类常量

public final class Constants
{
    /** GTR Timestamp formatter without micro-second */
    public static final FastDateFormat GTR_DATE_FORMAT 
                           = FastDateFormat.getInstance("yyyy-MM-dd'T'HH:mm:ss");
}

这是我目前的测试用例

@Test
public void testUpdateConfigDates() throws Exception
{
    // check if configDate in document is expected
    String docConfigDate = doc.getElementsByTagName("ConfigDate").item(0).getTextContent();
    assertEquals(docConfigDate, "2012-02-22T16:07:27Z");

    LOG.info("docConfigDate: " + docConfigDate);

    // variables
    Timestamp newConfigDate = Timestamp.valueOf("2009-07-29 13:24:11");

    // Mocking statics
    //PowerMockito.mockStatic(GTRConstants.class);

    // String configTimestampStr = GTR_DATE_FORMAT.format(configDate) + "Z";
    //common.setFinalStatic(GTRUtility.class.getDeclaredField("configTimestampStr"), "yyyy-MM-dd'T'HH:mm:ss" + "Z");
    common.setFinalStatic(Constants.class.getDeclaredField("GTR_DATE_FORMAT"), FastDateFormat.getInstance("yyyy-MM-dd'T'HH:mm:ss"));

    /* setFinalStatic ExceptionInInitializerError information below
     * 
     * Field#set(Object, Object) can be used to set static fields.
     * If you try to set the field of an uninitialized class, the JVM will first try to initialize the class.
     * If a failure occurs, then set will throw a ExceptionInInitializerError.
     */

    // execute method under test
    doc = gtrUtility.updateConfigDates(doc, newConfigDate);

    // verify expectations
    docConfigDate = doc.getElementsByTagName("ConfigDate").item(0).getTextContent();
    assertEquals(docConfigDate, "2009-07-29 13:24:11");
}

使用setFinalStatic solution posted here ,它可以成功地与我正在进行的其他一些测试一起使用。

public static void setFinalStatic(Field field, Object newValue) throws Exception
{
    field.setAccessible(true);
    // remove final modifier from field
    Field modifiersField = Field.class.getDeclaredField("modifiers");
    modifiersField.setAccessible(true);
    modifiersField.setInt(field, field.getModifiers() & ~Modifier.FINAL);
    field.set(null, newValue);
}

我尝试了几种解决方案:

  • 我的新方法和最新方法,使用setFinalStatic method
  • 使用PowerMockito和Mockito进行模拟常量。
  • 使用PowerMockito和Mockito模拟GTR_DATE_FORMAT实例。
  • 使用PowerMockito和Mockito模拟FastDateFormat。

所有这些都失败了。我认为setFinalStatic方法是最接近的,但我目前收到错误ExceptionInInitializerError(也从用户here和Oracle Doc here中讨论过)

1 个答案:

答案 0 :(得分:0)

事实证明,我的问题的答案是Constants文件正在执行一些初始化方法(在这种情况下:对象,来自数据库的safeGetTypeId(x)),我必须模拟。

当我意识到updateObjectId硬编码测试工作正常并且updateTimestamp使用常量类GTR_DATE_FORMAT时,我应该更快注意到这一点。

构造函数是空的,我不希望其他任何事情被执行。

这是一个极端而复杂的问题,所以我可能会删除这个问题,但如果它可以帮助某人将来我会离开它。