Как создать JarFile в памяти? - PullRequest
3 голосов
/ 06 июля 2011

Я пытаюсь написать функцию вроде:

public Map<String, Document> getTestXml(JarFile jarFile) {
    Map<String, Document> result = Maps.newHashMap();

    Enumeration<JarEntry> jarEntries = jarFile.getEntries();
    while (jarEntries.hasMoreElements()) {
        JarEntry jarEntry = jarEntries.nextElement();

        String name = jarEntry.getName();
        if (name.endsWith(".class") && !name.contains("$")) {
            String testClassName = name.replace(".class", "").replace("/", ".");
            String testXmlFilename = "TEST-" + testClassName + ".xml";

            InputStream testXmlInputStream = testJarFile.getInputStream(
                    testJarFile.getJarEntry(testXmlFilename));

            DocumentBuilderFactory documentBuilderFactory = DocumentBuilderFactory.newInstance();
            DocumentBuilder documentBuilder = documentBuilderFactory.newDocumentBuilder();
            Document testXmlDocument = documentBuilder.parse(testXmlInputStream);

            result.put(testClassName, testXmlDocument);
        }
    }

    return result;
}

И я хотел бы написать модульный тест, который на самом деле не создает JarFile в файловой системе. Я пытался найти способ создания объекта File в памяти, но не нашел ничего подобного. У кого-нибудь есть предложения?

Ответы [ 4 ]

4 голосов
/ 06 июля 2011

Вместо JarFile используйте JarInputStream.Для тестирования подключите JarInputStream к ByteArrayInputStream, загруженному с данными JAR в памяти, и при нормальной работе подключите его к входному потоку из файла.

1 голос
/ 06 июля 2011

File () объекты все живут в некотором пространстве имен файловой системы.Что дает вам два основных выбора:

1).Если вы используете O / S с файловой системой tempfs, создайте ее там.2).Используйте File.createTempFile () и установите атрибут delete-on-exit.

Обычный подход к созданию подкласса («public MemoryFile расширяет файл» ...) не работает, потому что File () объект не содержит методов для выполнения реального ввода-вывода, только для хранения имени объекта и выполнения нескольких операций файловой системы.

0 голосов
/ 06 июля 2011

Вам необходимо взглянуть на ByteArrayOutputStream и ByteArrayInputStream . Это потоковые объекты в памяти в Java. Используйте их, и ничего не будет записано на диск.

0 голосов
/ 06 июля 2011

Вы можете использовать EasyMock для создания фиктивного объекта класса JarFile.Для фиктивного объекта вы указываете, какие методы вызываются в тесте, а какие возвращаемые значения, без необходимости фактически создавать JAR-файл в файловой системе.

Затем вызовите ваш метод getTestXml () с вашим фиктивным JarFileпример.

Нужно некоторое время, чтобы привыкнуть к нему, но тогда вы увидите, что оно того стоит.

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

public class JarFileUser {
  public Map<String, Document> getTestXml(JarFile jarFile) throws IOException, ParserConfigurationException, SAXException {
    Map<String, Document> result = new HashMap<String, Document>();

    Enumeration<JarEntry> jarEntries = jarFile.entries();
    while (jarEntries.hasMoreElements()) {
      JarEntry jarEntry = jarEntries.nextElement();

      String name = jarEntry.getName();
      if (name.endsWith(".class") && !name.contains("$")) {
        String testClassName = name.replace(".class", "").replace("/", ".");
        String testXmlFilename = "TEST-" + testClassName + ".xml";

        InputStream testXmlInputStream = jarFile.getInputStream(jarFile.getJarEntry(testXmlFilename));

        DocumentBuilderFactory documentBuilderFactory = DocumentBuilderFactory.newInstance();
        DocumentBuilder documentBuilder = documentBuilderFactory.newDocumentBuilder();
        Document testXmlDocument = documentBuilder.parse(testXmlInputStream);

        result.put(testClassName, testXmlDocument);
      }
    }

    return result;
  }
}

Вот тест с EasyMock:

public class JarFileUserTest {

  private JarFile mockJarFile;
  private Enumeration<JarEntry> mockJarEntries;

  private JarFileUser jarFileUser;
  private JarEntry first;
  private JarEntry second;
  private JarEntry firstXml;

  @Before
  public void setUp() throws Exception {
    jarFileUser = new JarFileUser();

    // Create a mock for the JarFile parameter
    mockJarFile = createMock(JarFile.class);

    // User Vector to provide an Enumeration of JarEntry-Instances 
    Vector<JarEntry> entries = new Vector<JarEntry>();
    first = createMock(JarEntry.class);
    second = createMock(JarEntry.class);

    entries.add(first);
    entries.add(second);

    expect(first.getName()).andReturn("mocktest.JarFileUser.class");
    expect(second.getName()).andReturn("mocktest.Ignore$Me.class");

    mockJarEntries = entries.elements();

    expect(mockJarFile.entries()).andReturn(mockJarEntries);

    // JarEntry for the XML file
    firstXml = createMock(JarEntry.class);

    expect(mockJarFile.getJarEntry("TEST-mocktest.JarFileUser.xml")).andReturn(firstXml);

    // XML contents
    ByteArrayInputStream is = new ByteArrayInputStream("<test>This is a test.</test>".getBytes("UTF-8"));
    expect(mockJarFile.getInputStream(firstXml)).andReturn(is);

    replay(mockJarFile);
    replay(first);
    replay(second);
    replay(firstXml);
  }

  @Test
  public void testGetTestXml() throws IOException, ParserConfigurationException, SAXException {
    Map<String, Document> map = jarFileUser.getTestXml(mockJarFile);
    verify(mockJarFile);
    verify(first);
    verify(second);
    verify(firstXml);

    assertEquals(1, map.size());
    Document doc = map.get("mocktest.JarFileUser");
    assertNotNull(doc);
    final Element root = (Element) doc.getDocumentElement();
    assertNotNull(root);
    assertEquals("test", root.getNodeName());
    assertEquals("This is a test.", root.getTextContent());
  }

}

Примечание по дополнительным библиотекам JarFile - это класс, а не интерфейс, поэтому в соответствии с Документациями по установке EasyMock у вас должно быть Objenesis и cglib в вашем классе.

...