diff --git a/src/main/java/com/exactpro/th2/common/schema/factory/CommonFactory.java b/src/main/java/com/exactpro/th2/common/schema/factory/CommonFactory.java index 8fe4353c3..a49f1362f 100644 --- a/src/main/java/com/exactpro/th2/common/schema/factory/CommonFactory.java +++ b/src/main/java/com/exactpro/th2/common/schema/factory/CommonFactory.java @@ -1,5 +1,5 @@ /* - * Copyright 2020-2023 Exactpro (Exactpro Systems Limited) + * Copyright 2020-2024 Exactpro (Exactpro Systems Limited) * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at @@ -61,6 +61,7 @@ import java.util.Arrays; import java.util.Base64; import java.util.HashMap; +import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Objects; @@ -465,10 +466,23 @@ public Set getDictionaryAliases() { } try (Stream files = Files.list(dictionaryFolder)) { - return files + List fileList = files .filter(Files::isRegularFile) - .map(dictionary -> FilenameUtils.removeExtension(dictionary.getFileName().toString())) - .collect(Collectors.toSet()); + .collect(Collectors.toList()); + + Set aliasSet = new HashSet<>(); + Map> duplicates = new HashMap<>(); + for (Path path : fileList) { + String alias = FilenameUtils.removeExtension(path.getFileName().toString()).toLowerCase(); + if (!aliasSet.add(alias)) { + duplicates.computeIfAbsent(alias, (ignore) -> new HashSet<>()) + .add(path.getFileName().toString()); + } + } + if (!duplicates.isEmpty()) { + throw new IllegalStateException("Dictionary directory contains files with the same name in different cases, files: " + duplicates + ", path: " + dictionaryFolder.toAbsolutePath()); + } + return aliasSet; } } catch (IOException e) { throw new IllegalStateException("Can not get dictionaries aliases from path: " + dictionaryFolder.toAbsolutePath(), e); diff --git a/src/test/kotlin/com/exactpro/th2/common/schema/TestDictionaryLoad.kt b/src/test/kotlin/com/exactpro/th2/common/schema/TestDictionaryLoad.kt index 76fae29fa..172299b9a 100644 --- a/src/test/kotlin/com/exactpro/th2/common/schema/TestDictionaryLoad.kt +++ b/src/test/kotlin/com/exactpro/th2/common/schema/TestDictionaryLoad.kt @@ -17,85 +17,249 @@ package com.exactpro.th2.common.schema import com.exactpro.th2.common.schema.dictionary.DictionaryType import com.exactpro.th2.common.schema.factory.CommonFactory -import com.exactpro.th2.common.schema.factory.FactorySettings +import com.exactpro.th2.common.schema.util.ArchiveUtils +import org.apache.commons.lang3.RandomStringUtils import org.junit.jupiter.api.Assertions +import org.junit.jupiter.api.BeforeEach import org.junit.jupiter.api.Test +import org.junit.jupiter.api.io.TempDir +import org.junit.jupiter.params.ParameterizedTest +import org.junit.jupiter.params.provider.ValueSource import java.nio.file.Path +import kotlin.io.path.absolutePathString +import kotlin.io.path.createDirectories +import kotlin.io.path.writeBytes +import kotlin.io.path.writeText +import kotlin.test.assertEquals class TestDictionaryLoad { - @Test - fun `test file load dictionary`() { - val factory = CommonFactory.createFromArguments("-c", "src/test/resources/test_load_dictionaries") + @TempDir + lateinit var tempDir: Path + + @BeforeEach + fun beforeEach() { + writePrometheus(tempDir) + } - factory.readDictionary().use { - assert(String(it.readAllBytes()) == "test file") + //--//--//--readDictionary()--//--//--// + + @Test + fun `test read dictionary from old dictionary dir`() { + val content = writeDictionary(tempDir.resolve(Path.of("MAIN"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals(content, commonFactory.readDictionary().use { String(it.readAllBytes()) }) } } @Test - fun `test folder load dictionary`() { - val factory = CommonFactory.createFromArguments("-c", "src/test/resources/test_load_dictionaries") + fun `test read dictionary from old dictionary dir - file name mismatch`() { + writeDictionary(tempDir.resolve(Path.of("main"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + Assertions.assertThrows(IllegalStateException::class.java) { + commonFactory.readDictionary() + } + } + } + + @ParameterizedTest + @ValueSource(strings = ["MAIN", "main", "test-dictionary"]) + fun `test read dictionary from type dictionary dir`(fileName: String, ) { + val content = writeDictionary(tempDir.resolve(Path.of("dictionary", "main", fileName))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals(content, commonFactory.readDictionary().use { String(it.readAllBytes()) }) + } + } + + @ParameterizedTest + @ValueSource(strings = ["MAIN", "main", "test-dictionary"]) + fun `test read dictionary from type dictionary dir - dictionary name mismatch`( + fileName: String, + + ) { + writeDictionary(tempDir.resolve(Path.of("dictionary", "MAIN", fileName))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + Assertions.assertThrows(IllegalStateException::class.java) { + commonFactory.readDictionary() + } + } + } - factory.readDictionary(DictionaryType.LEVEL1).use { - assert(String(it.readAllBytes()) == "test file") + @ParameterizedTest + @ValueSource(strings = ["MAIN", "main", "MAIN.xml", "main.json"]) + fun `test read dictionary from alias dictionary dir`(fileName: String, ) { + val content = writeDictionary(tempDir.resolve(Path.of("dictionaries", fileName))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals(content, commonFactory.readDictionary().use { String(it.readAllBytes()) }) } } + //--//--//--readDictionary()--//--//--// + @Test - fun `test folder load dictionaries by alias`() { - val factory = CommonFactory.createFromArguments("-c", "src/test/resources/test_load_dictionaries") + fun `test read dictionary by type from old dictionary dir`() { + val content = writeDictionary(tempDir.resolve(Path.of("INCOMING"))) + writeDictionary(tempDir.resolve(Path.of("MAIN"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals( + content, + commonFactory.readDictionary(DictionaryType.INCOMING).use { String(it.readAllBytes()) }) + } + } - Assertions.assertDoesNotThrow { - factory.loadDictionary("test_alias_2").use { - assert(String(it.readAllBytes()) == "test file") + @Test + fun `test read dictionary by type from old dictionary dir - file name mismatch`() { + writeDictionary(tempDir.resolve(Path.of("incoming"))) + writeDictionary(tempDir.resolve(Path.of("MAIN"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + Assertions.assertThrows(IllegalStateException::class.java) { + commonFactory.readDictionary(DictionaryType.INCOMING) } } } - @Test - fun `test folder load all dictionary aliases`() { - val factory = CommonFactory.createFromArguments("-c", "src/test/resources/test_load_dictionaries") - val expectedNames = listOf("main", "test_alias_1", "test_alias_2", "test_alias_3", "test_alias_4") - val names = factory.dictionaryAliases - Assertions.assertEquals(5, names.size) - Assertions.assertTrue(names.containsAll(expectedNames)) + @ParameterizedTest + @ValueSource(strings = ["INCOMING", "incoming", "test-dictionary"]) + fun `test read dictionary by type from type dictionary dir`(fileName: String, ) { + val content = writeDictionary(tempDir.resolve(Path.of("dictionary", "incoming", fileName))) + writeDictionary(tempDir.resolve(Path.of("dictionary", "main", "MAIN"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals( + content, + commonFactory.readDictionary(DictionaryType.INCOMING).use { String(it.readAllBytes()) }) + } + } + + @ParameterizedTest + @ValueSource(strings = ["INCOMING", "incoming", "test-dictionary"]) + fun `test read dictionary by type from type dictionary dir - dictionary name mismatch`( + fileName: String, + + ) { + writeDictionary(tempDir.resolve(Path.of("dictionary", "INCOMING", fileName))) + writeDictionary(tempDir.resolve(Path.of("dictionary", "main", "MAIN"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + Assertions.assertThrows(IllegalStateException::class.java) { + commonFactory.readDictionary(DictionaryType.INCOMING) + } + } + } + + @ParameterizedTest + @ValueSource(strings = ["INCOMING", "incoming", "INCOMING.xml", "incoming.json"]) + fun `test read dictionary by type from alias dictionary dir`(fileName: String, ) { + val content = writeDictionary(tempDir.resolve(Path.of("dictionaries", fileName))) + writeDictionary(tempDir.resolve(Path.of("dictionaries", "MAIN"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals( + content, + commonFactory.readDictionary(DictionaryType.INCOMING).use { String(it.readAllBytes()) }) + } } + //--//--//--loadSingleDictionary()--//--//--// + @Test - fun `test folder load single dictionary from folder with several`() { - val factory = CommonFactory.createFromArguments("-c", "src/test/resources/test_load_dictionaries") + fun `test load single dictionary from alias dictionary dir`() { + val content = writeDictionary(tempDir.resolve(Path.of("dictionaries", "test-dictionary"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals(content, commonFactory.loadSingleDictionary().use { String(it.readAllBytes()) }) + } + } - Assertions.assertThrows(IllegalStateException::class.java) { - factory.loadSingleDictionary() + @Test + fun `test load single dictionary from alias dictionary dir - several dictionaries`() { + writeDictionary(tempDir.resolve(Path.of("dictionaries", "test-dictionary-1"))) + writeDictionary(tempDir.resolve(Path.of("dictionaries", "test-dictionary-2"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + Assertions.assertThrows(IllegalStateException::class.java) { + commonFactory.loadSingleDictionary() + } } } @Test - fun `test folder load single dictionary`() { - val customSettings = FactorySettings().apply { - prometheus = Path.of("src/test/resources/test_load_dictionaries/prometheus.json") - dictionaryAliasesDir = Path.of("src/test/resources/test_load_dictionaries/single_dictionary") + fun `test load single dictionary from alias dictionary dir - several dictionaries with name in different case`() { + writeDictionary(tempDir.resolve(Path.of("dictionaries", "test-dictionary"))) + writeDictionary(tempDir.resolve(Path.of("dictionaries", "TEST-DICTIONARY"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + Assertions.assertThrows(IllegalStateException::class.java) { + commonFactory.loadSingleDictionary() + } + } + } + + //--//--//--loadDictionary()--//--//--// + + @ParameterizedTest + @ValueSource(strings = ["TEST-ALIAS", "test-alias"]) + fun `test load dictionary by alias from alias dictionary dir`(fileName: String, ) { + val content = writeDictionary(tempDir.resolve(Path.of("dictionaries", fileName))) + writeDictionary(tempDir.resolve(Path.of("dictionaries", "test-dictionary"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals(content, commonFactory.loadDictionary("test-alias").use { String(it.readAllBytes()) }) + } + } + + @ParameterizedTest + @ValueSource(strings = ["TEST-DICTIONARY", "test-dictionary"]) + fun `test load dictionary by alias from alias dictionary dir - several dictionaries with name in different case`( + alias: String, + + ) { + writeDictionary(tempDir.resolve(Path.of("dictionaries", "test-dictionary"))) + writeDictionary(tempDir.resolve(Path.of("dictionaries", "TEST-DICTIONARY"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + Assertions.assertThrows(IllegalStateException::class.java) { + commonFactory.loadDictionary(alias) + } } - val customFactory = CommonFactory(customSettings) + } + + //--//--//--loadAliases--//--//--// - customFactory.loadSingleDictionary().use { - assert(String(it.readAllBytes()) == "test file") + @Test + fun `test dictionary aliases from alias dictionary dir`() { + val alias1 = "test-dictionary-1" + val alias2 = "TEST-DICTIONARY-2" + val file1 = "$alias1.xml" + val file2 = "$alias2.json" + writeDictionary(tempDir.resolve(Path.of("dictionaries", file1))) + writeDictionary(tempDir.resolve(Path.of("dictionaries", file2))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals(setOf(alias1, alias2.lowercase()), commonFactory.dictionaryAliases) } } @Test - fun `test folder load single dictionary by type as alias`() { - val customSettings = FactorySettings().apply { - prometheus = Path.of("src/test/resources/test_load_dictionaries/prometheus.json") - dictionaryTypesDir = Path.of("..") - dictionaryAliasesDir = Path.of("src/test/resources/test_load_dictionaries/dictionaries") + fun `test dictionary aliases from alias dictionary dir - empty directory`() { + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + assertEquals(emptySet(), commonFactory.dictionaryAliases) } - val customFactory = CommonFactory(customSettings) + } - customFactory.readDictionary().use { - assert(String(it.readAllBytes()) == "test file") + @Test + fun `test dictionary aliases from alias dictionary dir - several dictionaries with name in different case`() { + writeDictionary(tempDir.resolve(Path.of("dictionaries", "test-dictionary"))) + writeDictionary(tempDir.resolve(Path.of("dictionaries", "TEST-DICTIONARY"))) + CommonFactory.createFromArguments("-c", tempDir.absolutePathString()).use { commonFactory -> + Assertions.assertThrows(IllegalStateException::class.java) { + commonFactory.dictionaryAliases + } } } + //--//--//--Others--//--//--// + + private fun writePrometheus(cfgPath: Path) { + cfgPath.resolve("prometheus.json").writeText("{\"enabled\":false}") + } + + private fun writeDictionary(path: Path): String? { + val content = RandomStringUtils.randomAlphanumeric(10) + path.parent.createDirectories() + path.writeBytes(ArchiveUtils.getGzipBase64StringEncoder().encode(content)) + return content + } + } \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/dictionaries/main b/src/test/resources/test_load_dictionaries/dictionaries/main deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/dictionaries/main +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/dictionaries/test_alias_1.encoded b/src/test/resources/test_load_dictionaries/dictionaries/test_alias_1.encoded deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/dictionaries/test_alias_1.encoded +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/dictionaries/test_alias_2 b/src/test/resources/test_load_dictionaries/dictionaries/test_alias_2 deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/dictionaries/test_alias_2 +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/dictionaries/test_alias_3.encoded b/src/test/resources/test_load_dictionaries/dictionaries/test_alias_3.encoded deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/dictionaries/test_alias_3.encoded +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/dictionaries/test_alias_4 b/src/test/resources/test_load_dictionaries/dictionaries/test_alias_4 deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/dictionaries/test_alias_4 +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/level1/test_dictionary.encoded b/src/test/resources/test_load_dictionaries/level1/test_dictionary.encoded deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/level1/test_dictionary.encoded +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/prometheus.json b/src/test/resources/test_load_dictionaries/prometheus.json deleted file mode 100644 index c994fef1c..000000000 --- a/src/test/resources/test_load_dictionaries/prometheus.json +++ /dev/null @@ -1,3 +0,0 @@ -{ - "enabled": false -} \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/single_dictionary/test_alias_1.encoded b/src/test/resources/test_load_dictionaries/single_dictionary/test_alias_1.encoded deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/single_dictionary/test_alias_1.encoded +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/test_dictionary_LEVEL1.encoded b/src/test/resources/test_load_dictionaries/test_dictionary_LEVEL1.encoded deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/test_dictionary_LEVEL1.encoded +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file diff --git a/src/test/resources/test_load_dictionaries/test_dictionary_MAIN.encoded b/src/test/resources/test_load_dictionaries/test_dictionary_MAIN.encoded deleted file mode 100644 index 0f3ff0936..000000000 --- a/src/test/resources/test_load_dictionaries/test_dictionary_MAIN.encoded +++ /dev/null @@ -1 +0,0 @@ -H4sIAAAAAAAAACtJLS5RSMvMSQUAwWtk8gkAAAA= \ No newline at end of file