From c6d2d173946a21721417ab6461257fc6e418bca2 Mon Sep 17 00:00:00 2001 From: Zvi Grinberg Date: Wed, 6 Mar 2024 00:19:27 +0200 Subject: [PATCH] test: mock maven interaction in Integration test To overcome bug of running dependency tree maven plugin in a github action workflow runner, with any combination of java + maven Signed-off-by: Zvi Grinberg --- .../com/redhat/exhort/impl/ExhortApiIT.java | 43 +++++++++++++++++-- .../providers/Java_Maven_Provider_Test.java | 6 ++- .../tst_manifests/it/maven/depTree.txt | 6 +++ 3 files changed, 50 insertions(+), 5 deletions(-) create mode 100644 src/test/resources/tst_manifests/it/maven/depTree.txt diff --git a/src/test/java/com/redhat/exhort/impl/ExhortApiIT.java b/src/test/java/com/redhat/exhort/impl/ExhortApiIT.java index a8d75c88..f6cf8f70 100644 --- a/src/test/java/com/redhat/exhort/impl/ExhortApiIT.java +++ b/src/test/java/com/redhat/exhort/impl/ExhortApiIT.java @@ -24,13 +24,15 @@ import com.redhat.exhort.api.ProviderReport; import com.redhat.exhort.providers.HelperExtension; import com.redhat.exhort.tools.Ecosystem; +import com.redhat.exhort.tools.Operations; import org.junit.jupiter.api.AfterAll; import org.junit.jupiter.api.BeforeAll; import org.junit.jupiter.api.Tag; -import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; import org.junit.jupiter.params.ParameterizedTest; import org.junit.jupiter.params.provider.EnumSource; +import org.mockito.MockedStatic; +import org.mockito.junit.jupiter.MockitoExtension; import java.io.IOException; import java.net.HttpURLConnection; @@ -38,14 +40,20 @@ import java.util.Map; import java.util.concurrent.ExecutionException; +import static com.redhat.exhort.providers.Java_Maven_Provider_Test.getOutputFileAndOverwriteItWithMock; import static org.junit.jupiter.api.Assertions.*; +import static org.mockito.ArgumentMatchers.any; +import static org.mockito.Mockito.mockStatic; @Tag("IntegrationTest") @ExtendWith(HelperExtension.class) +@ExtendWith(MockitoExtension.class) class ExhortApiIT extends ExhortTest { private static Api api; private static Map ecoSystemsManifestNames; + + private MockedStatic mockedOperations; @BeforeAll static void beforeAll() { api = new ExhortApi(); @@ -67,9 +75,21 @@ void Integration_Test_End_To_End_Stack_Analysis(Ecosystem.Type packageManager) t String manifestFileName = ecoSystemsManifestNames.get(packageManager.getType()); String pathToManifest = getFileFromResource(manifestFileName, "tst_manifests", "it", packageManager.getType(), manifestFileName); preparePythonEnvironment(packageManager); + // Github action runner with all maven and java versions seems to enter infinite loop in integration tests of MAVEN when runnig dependency maven plugin to produce verbose text dependenct tree format. + // locally it's not recreated with same versions + mockMavenDependencyTree(packageManager); + releaseStaticMock(packageManager); AnalysisReport analysisReportResult = api.stackAnalysis(pathToManifest).get(); handleJsonResponse(analysisReportResult,true); } + + private void releaseStaticMock(Ecosystem.Type packageManager) { + if(packageManager.equals(Ecosystem.Type.MAVEN)) { + this.mockedOperations.close(); + } + } + + @Tag("IntegrationTest") @ParameterizedTest @EnumSource(value = Ecosystem.Type.class, names = { "GOLANG", "MAVEN", "NPM", "PYTHON" }) @@ -77,6 +97,10 @@ void Integration_Test_End_To_End_Stack_Analysis_Mixed(Ecosystem.Type packageMana String manifestFileName = ecoSystemsManifestNames.get(packageManager.getType()); String pathToManifest = getFileFromResource(manifestFileName, "tst_manifests", "it", packageManager.getType(), manifestFileName); preparePythonEnvironment(packageManager); + // Github action runner with all maven and java versions seems to enter infinite loop in integration tests of MAVEN when runnig dependency maven plugin to produce verbose text dependenct tree format. + // locally it's not recreated with same versions + mockMavenDependencyTree(packageManager); + releaseStaticMock(packageManager); AnalysisReport analysisReportJson = api.stackAnalysisMixed(pathToManifest).get().json; String analysisReportHtml = new String(api.stackAnalysisMixed(pathToManifest).get().html); handleJsonResponse(analysisReportJson,true); @@ -90,6 +114,10 @@ void Integration_Test_End_To_End_Stack_Analysis_Html(Ecosystem.Type packageManag String manifestFileName = ecoSystemsManifestNames.get(packageManager.getType()); String pathToManifest = getFileFromResource(manifestFileName, "tst_manifests", "it", packageManager.getType(), manifestFileName); preparePythonEnvironment(packageManager); + // Github action runner with all maven and java versions seems to enter infinite loop in integration tests of MAVEN when runnig dependency maven plugin to produce verbose text dependenct tree format. + // locally it's not recreated with same versions + mockMavenDependencyTree(packageManager); + releaseStaticMock(packageManager); String analysisReportHtml = new String(api.stackAnalysisHtml(pathToManifest).get()); handleHtmlResponse(analysisReportHtml); } @@ -101,7 +129,7 @@ void Integration_Test_End_To_End_Stack_Analysis_Html(Ecosystem.Type packageManag void Integration_Test_End_To_End_Component_Analysis(Ecosystem.Type packageManager) throws IOException, ExecutionException, InterruptedException { String manifestFileName = ecoSystemsManifestNames.get(packageManager.getType()); byte[] manifestContent = getStringFromFile("tst_manifests", "it", packageManager.getType(), manifestFileName).getBytes(); - preparePythonEnvironment(packageManager); + preparePythonEnvironment(packageManager); AnalysisReport analysisReportResult = api.componentAnalysis(manifestFileName,manifestContent).get(); handleJsonResponse(analysisReportResult,false); } @@ -155,7 +183,16 @@ private void handleHtmlResponse(String analysisReportHtml) throws JsonProcessing assertTrue(status.get("ok").asBoolean(false)); } - + private void mockMavenDependencyTree(Ecosystem.Type packageManager) throws IOException { + if(packageManager.equals(Ecosystem.Type.MAVEN)) { + mockedOperations = mockStatic(Operations.class); + String depTree; + try (var is = getResourceAsStreamDecision(getClass(), new String [] { "tst_manifests", "it","maven", "depTree.txt"})) { + depTree = new String(is.readAllBytes()); + } + mockedOperations.when(() -> Operations.runProcess(any(),any())).thenAnswer(invocationOnMock -> getOutputFileAndOverwriteItWithMock(depTree, invocationOnMock, "-DoutputFile")); + } + } } diff --git a/src/test/java/com/redhat/exhort/providers/Java_Maven_Provider_Test.java b/src/test/java/com/redhat/exhort/providers/Java_Maven_Provider_Test.java index 593dc641..2fbde9b0 100644 --- a/src/test/java/com/redhat/exhort/providers/Java_Maven_Provider_Test.java +++ b/src/test/java/com/redhat/exhort/providers/Java_Maven_Provider_Test.java @@ -41,7 +41,9 @@ @ExtendWith(HelperExtension.class) @ExtendWith(MockitoExtension.class) -class Java_Maven_Provider_Test extends ExhortTest { +public class Java_Maven_Provider_Test extends ExhortTest { + + // private static System.Logger log = System.getLogger("Java_Maven_Provider_Test"); // test folder are located at src/test/resources/tst_manifests // each folder should contain: @@ -100,7 +102,7 @@ void test_the_provideStack(String testFolder) throws IOException, InterruptedExc } - private static String getOutputFileAndOverwriteItWithMock(String outputFileContent, InvocationOnMock invocationOnMock,String parameterPrefix) throws IOException { + public static String getOutputFileAndOverwriteItWithMock(String outputFileContent, InvocationOnMock invocationOnMock,String parameterPrefix) throws IOException { String[] rawArguments = (String[]) invocationOnMock.getRawArguments()[0]; Optional outputFileArg = Arrays.stream(rawArguments).filter(arg -> arg!= null && arg.startsWith(parameterPrefix)).findFirst(); String outputFilePath=null; diff --git a/src/test/resources/tst_manifests/it/maven/depTree.txt b/src/test/resources/tst_manifests/it/maven/depTree.txt new file mode 100644 index 00000000..42751313 --- /dev/null +++ b/src/test/resources/tst_manifests/it/maven/depTree.txt @@ -0,0 +1,6 @@ +pom-with-deps-no-ignore:pom-with-dependency-not-ignored-for-tests:jar:0.0.1 ++- log4j:log4j:jar:1.2.17:compile ++- org.projectlombok:lombok:jar:1.16.6:compile +\- com.fasterxml.jackson.core:jackson-databind:jar:2.14.0:compile + +- com.fasterxml.jackson.core:jackson-annotations:jar:2.14.0:compile + \- com.fasterxml.jackson.core:jackson-core:jar:2.14.0:compile