2021-06-19 18:10:26 +02:00
|
|
|
package prog8tests
|
|
|
|
|
2021-07-11 17:32:29 +02:00
|
|
|
import prog8tests.helpers.*
|
2021-08-01 17:24:12 +02:00
|
|
|
import org.hamcrest.MatcherAssert.assertThat
|
|
|
|
import org.hamcrest.Matchers.equalTo
|
|
|
|
import org.hamcrest.Matchers.containsString
|
2021-07-18 19:02:47 +02:00
|
|
|
import org.junit.jupiter.api.Test
|
|
|
|
import org.junit.jupiter.api.TestInstance
|
2021-08-01 17:24:12 +02:00
|
|
|
import org.junit.jupiter.api.BeforeEach
|
2021-08-01 15:15:54 +02:00
|
|
|
import org.junit.jupiter.api.Disabled
|
|
|
|
import org.junit.jupiter.api.Nested
|
2021-07-18 19:02:47 +02:00
|
|
|
import org.junit.jupiter.api.assertThrows
|
2021-06-21 12:02:36 +02:00
|
|
|
import kotlin.io.path.*
|
|
|
|
|
2021-06-19 18:10:26 +02:00
|
|
|
import prog8.ast.Program
|
|
|
|
import prog8.parser.ParseError
|
|
|
|
|
2021-08-02 09:15:11 +02:00
|
|
|
import prog8.compiler.ModuleImporter
|
2021-08-01 17:24:12 +02:00
|
|
|
import kotlin.test.assertContains
|
2021-07-18 19:02:47 +02:00
|
|
|
|
2021-06-19 18:10:26 +02:00
|
|
|
|
2021-06-19 20:27:04 +02:00
|
|
|
@TestInstance(TestInstance.Lifecycle.PER_CLASS)
|
2021-06-19 18:10:26 +02:00
|
|
|
class TestModuleImporter {
|
2021-07-18 19:02:47 +02:00
|
|
|
private val count = listOf("1st", "2nd", "3rd", "4th", "5th")
|
|
|
|
|
2021-08-01 22:47:11 +02:00
|
|
|
private lateinit var program: Program
|
2021-08-01 17:24:12 +02:00
|
|
|
@BeforeEach
|
|
|
|
fun beforeEach() {
|
2021-08-01 22:47:11 +02:00
|
|
|
program = Program("foo", DummyFunctions, DummyMemsizer)
|
2021-08-01 17:24:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private fun makeImporter(vararg searchIn: String): ModuleImporter = makeImporter(searchIn.asList())
|
|
|
|
|
2021-08-01 22:47:11 +02:00
|
|
|
private fun makeImporter(searchIn: Iterable<String>) =
|
|
|
|
ModuleImporter(program, "blah", searchIn.toList())
|
2021-08-01 17:24:12 +02:00
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Nested
|
|
|
|
inner class Constructor {
|
2021-07-18 19:02:47 +02:00
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Test
|
|
|
|
@Disabled("TODO: invalid entries in search list")
|
|
|
|
fun testInvalidEntriesInSearchList() {}
|
2021-07-18 19:02:47 +02:00
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Test
|
|
|
|
@Disabled("TODO: literal duplicates in search list")
|
|
|
|
fun testLiteralDuplicatesInSearchList() {}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
@Disabled("TODO: factual duplicates in search list")
|
|
|
|
fun testFactualDuplicatesInSearchList() {}
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Nested
|
|
|
|
inner class ImportModule {
|
2021-06-19 20:45:37 +02:00
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Nested
|
|
|
|
inner class WithInvalidPath {
|
|
|
|
@Test
|
|
|
|
fun testNonexisting() {
|
|
|
|
val dirRel = assumeDirectory(".", workingDir.relativize(fixturesDir))
|
2021-08-01 17:24:12 +02:00
|
|
|
val importer = makeImporter(dirRel.invariantSeparatorsPathString)
|
2021-08-01 15:15:54 +02:00
|
|
|
val srcPathRel = assumeNotExists(dirRel, "i_do_not_exist")
|
|
|
|
val srcPathAbs = srcPathRel.absolute()
|
2021-06-19 20:45:37 +02:00
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
assertThrows<NoSuchFileException> { importer.importModule(srcPathRel) }
|
|
|
|
.let {
|
|
|
|
assertThat(
|
|
|
|
".file should be normalized",
|
2021-08-01 17:24:12 +02:00
|
|
|
"${it.file}", equalTo("${it.file.normalize()}")
|
2021-08-01 15:15:54 +02:00
|
|
|
)
|
|
|
|
assertThat(
|
|
|
|
".file should point to specified path",
|
2021-08-01 17:24:12 +02:00
|
|
|
it.file.absolutePath, equalTo("${srcPathAbs.normalize()}")
|
2021-08-01 15:15:54 +02:00
|
|
|
)
|
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(1))
|
2021-08-01 15:15:54 +02:00
|
|
|
|
|
|
|
assertThrows<NoSuchFileException> { importer.importModule(srcPathAbs) }
|
|
|
|
.let {
|
|
|
|
assertThat(
|
|
|
|
".file should be normalized",
|
2021-08-01 17:24:12 +02:00
|
|
|
"${it.file}", equalTo("${it.file.normalize()}")
|
2021-08-01 15:15:54 +02:00
|
|
|
)
|
|
|
|
assertThat(
|
|
|
|
".file should point to specified path",
|
2021-08-01 17:24:12 +02:00
|
|
|
it.file.absolutePath, equalTo("${srcPathAbs.normalize()}")
|
2021-08-01 15:15:54 +02:00
|
|
|
)
|
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(1))
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
2021-06-19 20:45:37 +02:00
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Test
|
|
|
|
fun testDirectory() {
|
2021-08-01 22:47:11 +02:00
|
|
|
val srcPathRel = assumeDirectory(workingDir.relativize(fixturesDir))
|
2021-08-01 15:15:54 +02:00
|
|
|
val srcPathAbs = srcPathRel.absolute()
|
2021-08-01 22:47:11 +02:00
|
|
|
val searchIn = Path(".", "$srcPathRel").invariantSeparatorsPathString
|
|
|
|
val importer = makeImporter(searchIn)
|
2021-08-01 15:15:54 +02:00
|
|
|
|
|
|
|
assertThrows<AccessDeniedException> { importer.importModule(srcPathRel) }
|
|
|
|
.let {
|
|
|
|
assertThat(
|
|
|
|
".file should be normalized",
|
2021-08-01 17:24:12 +02:00
|
|
|
"${it.file}", equalTo("${it.file.normalize()}")
|
2021-08-01 15:15:54 +02:00
|
|
|
)
|
|
|
|
assertThat(
|
|
|
|
".file should point to specified path",
|
2021-08-01 17:24:12 +02:00
|
|
|
it.file.absolutePath, equalTo("${srcPathAbs.normalize()}")
|
2021-08-01 15:15:54 +02:00
|
|
|
)
|
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(1))
|
2021-08-01 15:15:54 +02:00
|
|
|
|
|
|
|
assertThrows<AccessDeniedException> { importer.importModule(srcPathAbs) }
|
|
|
|
.let {
|
|
|
|
assertThat(
|
|
|
|
".file should be normalized",
|
2021-08-01 17:24:12 +02:00
|
|
|
"${it.file}", equalTo("${it.file.normalize()}")
|
2021-08-01 15:15:54 +02:00
|
|
|
)
|
|
|
|
assertThat(
|
|
|
|
".file should point to specified path",
|
2021-08-01 17:24:12 +02:00
|
|
|
it.file.absolutePath, equalTo("${srcPathAbs.normalize()}")
|
2021-08-01 15:15:54 +02:00
|
|
|
)
|
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(1))
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
2021-06-19 18:10:26 +02:00
|
|
|
}
|
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Nested
|
|
|
|
inner class WithValidPath {
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun testAbsolute() {
|
|
|
|
val searchIn = listOf(
|
|
|
|
Path(".").div(workingDir.relativize(fixturesDir)), // we do want a dot "." in front
|
|
|
|
).map { it.invariantSeparatorsPathString }
|
2021-08-01 17:24:12 +02:00
|
|
|
val importer = makeImporter(searchIn)
|
2021-08-01 15:15:54 +02:00
|
|
|
val fileName = "simple_main.p8"
|
|
|
|
val path = assumeReadableFile(searchIn[0], fileName)
|
|
|
|
|
|
|
|
val module = importer.importModule(path.absolute())
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(2))
|
|
|
|
assertContains(program.modules, module)
|
|
|
|
assertThat(module.program, equalTo(program))
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun testRelativeToWorkingDir() {
|
|
|
|
val searchIn = listOf(
|
|
|
|
Path(".").div(workingDir.relativize(fixturesDir)), // we do want a dot "." in front
|
|
|
|
).map { it.invariantSeparatorsPathString }
|
2021-08-01 17:24:12 +02:00
|
|
|
val importer = makeImporter(searchIn)
|
2021-08-01 15:15:54 +02:00
|
|
|
val fileName = "simple_main.p8"
|
|
|
|
val path = assumeReadableFile(searchIn[0], fileName)
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat("sanity check: path should NOT be absolute", path.isAbsolute, equalTo(false))
|
2021-08-01 15:15:54 +02:00
|
|
|
|
|
|
|
val module = importer.importModule(path)
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(2))
|
|
|
|
assertContains(program.modules, module)
|
|
|
|
assertThat(module.program, equalTo(program))
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
fun testRelativeTo1stDirInSearchList() {
|
2021-08-01 17:24:12 +02:00
|
|
|
val searchIn = Path(".")
|
|
|
|
.div(workingDir.relativize(fixturesDir))
|
|
|
|
.invariantSeparatorsPathString
|
|
|
|
val importer = makeImporter(searchIn)
|
2021-08-01 15:15:54 +02:00
|
|
|
val fileName = "simple_main.p8"
|
|
|
|
val path = Path(".", fileName)
|
2021-08-01 17:24:12 +02:00
|
|
|
assumeReadableFile(searchIn, path)
|
2021-08-01 15:15:54 +02:00
|
|
|
|
|
|
|
val module = importer.importModule(path)
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(2))
|
|
|
|
assertContains(program.modules, module)
|
|
|
|
assertThat(module.program, equalTo(program))
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
2021-06-19 20:27:04 +02:00
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Test
|
|
|
|
@Disabled("TODO: relative to 2nd in search list")
|
|
|
|
fun testRelativeTo2ndDirInSearchList() {}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
@Disabled("TODO: ambiguous - 2 or more really different candidates")
|
|
|
|
fun testAmbiguousCandidates() {}
|
|
|
|
|
|
|
|
@Nested
|
|
|
|
inner class WithBadFile {
|
|
|
|
@Test
|
|
|
|
fun testWithSyntaxError() {
|
2021-08-01 17:24:12 +02:00
|
|
|
val searchIn = assumeDirectory("./", workingDir.relativize(fixturesDir))
|
|
|
|
val importer = makeImporter(searchIn.invariantSeparatorsPathString)
|
2021-08-01 15:15:54 +02:00
|
|
|
val srcPath = assumeReadableFile(fixturesDir, "file_with_syntax_error.p8")
|
|
|
|
|
|
|
|
val act = { importer.importModule(srcPath) }
|
|
|
|
|
|
|
|
repeat(2) { n ->
|
|
|
|
assertThrows<ParseError>(count[n] + " call") { act() }.let {
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(it.position.file, equalTo(srcPath.absolutePathString()))
|
|
|
|
assertThat("line; should be 1-based", it.position.line, equalTo(2))
|
|
|
|
assertThat("startCol; should be 0-based", it.position.startCol, equalTo(6))
|
|
|
|
assertThat("endCol; should be 0-based", it.position.endCol, equalTo(6))
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(1))
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
2021-08-01 15:15:54 +02:00
|
|
|
|
|
|
|
@Test
|
2021-08-01 22:47:11 +02:00
|
|
|
fun testImportingFileWithSyntaxError_once() {
|
|
|
|
doTestImportingFileWithSyntaxError(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
@Disabled("TODO: module that imports faulty module should not be kept in Program.modules")
|
|
|
|
fun testImportingFileWithSyntaxError_twice() {
|
|
|
|
doTestImportingFileWithSyntaxError(2)
|
|
|
|
}
|
|
|
|
|
|
|
|
private fun doTestImportingFileWithSyntaxError(repetitions: Int) {
|
2021-08-01 17:24:12 +02:00
|
|
|
val searchIn = assumeDirectory("./", workingDir.relativize(fixturesDir))
|
|
|
|
val importer = makeImporter(searchIn.invariantSeparatorsPathString)
|
2021-08-01 15:15:54 +02:00
|
|
|
val importing = assumeReadableFile(fixturesDir, "import_file_with_syntax_error.p8")
|
|
|
|
val imported = assumeReadableFile(fixturesDir, "file_with_syntax_error.p8")
|
|
|
|
|
|
|
|
val act = { importer.importModule(importing) }
|
|
|
|
|
2021-08-01 22:47:11 +02:00
|
|
|
repeat(repetitions) { n ->
|
2021-08-01 15:15:54 +02:00
|
|
|
assertThrows<ParseError>(count[n] + " call") { act() }.let {
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(it.position.file, equalTo(imported.absolutePathString()))
|
|
|
|
assertThat("line; should be 1-based", it.position.line, equalTo(2))
|
|
|
|
assertThat("startCol; should be 0-based", it.position.startCol, equalTo(6))
|
|
|
|
assertThat("endCol; should be 0-based", it.position.endCol, equalTo(6))
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
// TODO("assertThat(program.modules.size, equalTo(2))")
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
2021-08-01 15:15:54 +02:00
|
|
|
|
2021-06-21 12:02:36 +02:00
|
|
|
}
|
2021-06-19 20:27:04 +02:00
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Nested
|
|
|
|
inner class ImportLibraryModule {
|
|
|
|
@Nested
|
|
|
|
inner class WithInvalidName {
|
|
|
|
@Test
|
|
|
|
fun testWithNonExistingName() {
|
2021-08-01 17:24:12 +02:00
|
|
|
val searchIn = assumeDirectory("./", workingDir.relativize(fixturesDir))
|
|
|
|
val importer = makeImporter(searchIn.invariantSeparatorsPathString)
|
2021-08-01 15:15:54 +02:00
|
|
|
val filenameNoExt = assumeNotExists(fixturesDir, "i_do_not_exist").name
|
|
|
|
val filenameWithExt = assumeNotExists(fixturesDir, "i_do_not_exist.p8").name
|
|
|
|
|
|
|
|
repeat(2) { n ->
|
|
|
|
assertThrows<NoSuchFileException>(count[n] + " call / NO .p8 extension")
|
|
|
|
{ importer.importLibraryModule(filenameNoExt) }.let {
|
|
|
|
assertThat(it.message!!, containsString(filenameWithExt))
|
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(1))
|
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
assertThrows<NoSuchFileException>(count[n] + " call / with .p8 extension")
|
|
|
|
{ importer.importLibraryModule(filenameWithExt) }.let {
|
|
|
|
assertThat(it.message!!, containsString(filenameWithExt))
|
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(program.modules.size, equalTo(1))
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
2021-06-19 20:27:04 +02:00
|
|
|
}
|
|
|
|
|
2021-08-01 15:15:54 +02:00
|
|
|
@Nested
|
|
|
|
inner class WithValidName {
|
|
|
|
@Nested
|
|
|
|
inner class WithBadFile {
|
|
|
|
@Test
|
|
|
|
fun testWithSyntaxError() {
|
2021-08-01 17:24:12 +02:00
|
|
|
val searchIn = assumeDirectory("./", workingDir.relativize(fixturesDir))
|
|
|
|
val importer = makeImporter(searchIn.invariantSeparatorsPathString)
|
2021-08-01 15:15:54 +02:00
|
|
|
val srcPath = assumeReadableFile(fixturesDir, "file_with_syntax_error.p8")
|
|
|
|
|
|
|
|
repeat(2) { n ->
|
|
|
|
assertThrows<ParseError>(count[n] + " call")
|
2021-08-01 17:24:12 +02:00
|
|
|
{ importer.importLibraryModule(srcPath.nameWithoutExtension) }.let {
|
|
|
|
assertThat(it.position.file, equalTo(srcPath.absolutePathString()))
|
|
|
|
assertThat("line; should be 1-based", it.position.line, equalTo(2))
|
|
|
|
assertThat("startCol; should be 0-based", it.position.startCol, equalTo(6))
|
|
|
|
assertThat("endCol; should be 0-based", it.position.endCol, equalTo(6))
|
|
|
|
}
|
|
|
|
assertThat(program.modules.size, equalTo(1))
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-01 16:26:27 +02:00
|
|
|
|
|
|
|
private fun doTestImportingFileWithSyntaxError(repetitions: Int) {
|
2021-08-01 17:24:12 +02:00
|
|
|
val searchIn = assumeDirectory("./", workingDir.relativize(fixturesDir))
|
|
|
|
val importer = makeImporter(searchIn.invariantSeparatorsPathString)
|
2021-08-01 15:15:54 +02:00
|
|
|
val importing = assumeReadableFile(fixturesDir, "import_file_with_syntax_error.p8")
|
|
|
|
val imported = assumeReadableFile(fixturesDir, "file_with_syntax_error.p8")
|
|
|
|
|
|
|
|
val act = { importer.importLibraryModule(importing.nameWithoutExtension) }
|
|
|
|
|
2021-08-01 16:26:27 +02:00
|
|
|
repeat(repetitions) { n ->
|
2021-08-01 15:15:54 +02:00
|
|
|
assertThrows<ParseError>(count[n] + " call") { act() }.let {
|
2021-08-01 17:24:12 +02:00
|
|
|
assertThat(it.position.file, equalTo(imported.normalize().absolutePathString()))
|
|
|
|
assertThat("line; should be 1-based", it.position.line, equalTo(2))
|
|
|
|
assertThat("startCol; should be 0-based", it.position.startCol, equalTo(6))
|
|
|
|
assertThat("endCol; should be 0-based", it.position.endCol, equalTo(6))
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
2021-08-01 17:24:12 +02:00
|
|
|
// TODO("assertThat(program.modules.size, equalTo(1))")
|
2021-08-01 15:15:54 +02:00
|
|
|
}
|
|
|
|
}
|
2021-08-01 16:26:27 +02:00
|
|
|
|
|
|
|
@Test
|
|
|
|
fun testImportingFileWithSyntaxError_once() {
|
|
|
|
doTestImportingFileWithSyntaxError(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
@Disabled("TODO: module that imports faulty module should not be kept in Program.modules")
|
|
|
|
fun testImportingFileWithSyntaxError_twice() {
|
|
|
|
doTestImportingFileWithSyntaxError(2)
|
|
|
|
}
|
2021-07-18 19:02:47 +02:00
|
|
|
}
|
2021-06-19 18:10:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|