-
Notifications
You must be signed in to change notification settings - Fork 61
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Dumps the ported typer tests and adds appropriate testing mechanisms. #1574
base: v1
Are you sure you want to change the base?
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,236 @@ | ||
package org.partiql.planner | ||
|
||
import com.amazon.ionelement.api.loadSingleElement | ||
import org.partiql.parser.PartiQLParser | ||
import org.partiql.plan.Statement | ||
import org.partiql.plan.debug.PlanPrinter | ||
import org.partiql.planner.catalog.Catalog | ||
import org.partiql.planner.catalog.Name | ||
import org.partiql.planner.catalog.Session | ||
import org.partiql.planner.internal.TestCatalog | ||
import org.partiql.planner.test.Test | ||
import org.partiql.planner.util.ProblemCollector | ||
import org.partiql.plugins.local.toStaticType | ||
import org.partiql.types.PType | ||
|
||
/** | ||
* This is a port of the original assertion logic from [org.partiql.planner.internal.typer.PlanTyperTestsPorted]. | ||
* | ||
* @see Success | ||
* @see org.partiql.planner.internal.typer.PlanTyperTestsPorted.TestCase.SuccessTestCase | ||
* @see Failure | ||
* @see org.partiql.planner.internal.typer.PlanTyperTestsPorted.TestCase.ErrorTestCase | ||
*/ | ||
abstract class TyperTest( | ||
private val _name: String, | ||
open var expectedType: PType?, | ||
var statement: String, | ||
var catalog: String, | ||
var directory: List<String>, | ||
) : Test { | ||
|
||
val parser = PartiQLParser.default() | ||
val planner = PartiQLPlanner.builder().signal().build() | ||
|
||
override fun getName(): String { | ||
return _name | ||
} | ||
|
||
override fun toString(): String { | ||
return getName() | ||
} | ||
|
||
val catalogs: List<Catalog> by lazy { | ||
// Make a map from catalog name to tables. | ||
val inputStream = this::class.java.getResourceAsStream("/resource_path.txt")!! | ||
val map = mutableMapOf<String, MutableList<Pair<Name, PType>>>() | ||
inputStream.reader().readLines().forEach { path -> | ||
if (path.startsWith("catalogs/default")) { | ||
val schema = this::class.java.getResourceAsStream("/$path")!! | ||
val ion = loadSingleElement(schema.reader().readText()) | ||
val staticType = ion.toStaticType() | ||
val steps = path.substring(0, path.length - 4).split('/').drop(2) // drop the catalogs/default | ||
val catalogName = steps.first() | ||
// args | ||
val name = Name.of(steps.drop(1)) | ||
val ptype = PType.fromStaticType(staticType) | ||
if (map.containsKey(catalogName)) { | ||
map[catalogName]!!.add(name to ptype) | ||
} else { | ||
map[catalogName] = mutableListOf(name to ptype) | ||
} | ||
} | ||
} | ||
// Make a catalogs list | ||
map.map { (catalog, tables) -> | ||
TestCatalog.builder() | ||
.name(catalog) | ||
.apply { | ||
for ((name, schema) in tables) { | ||
createTable(name, schema) | ||
} | ||
} | ||
.build() | ||
} | ||
} | ||
|
||
/** | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Could provide the description of the test format in some other more visible location such as https://github.com/johnedquinn/partiql-lang-jvm/blob/v1-tests-dump/partiql-planner/src/testFixtures/resources/README.adoc. Something like https://github.com/partiql/partiql-tests/blob/main/docs/partiql-tests-schema-proposal.md#evaluation-tests, could make it easier to add future tests. |
||
* This represents a successful test case. | ||
* | ||
* The serialization of a test case looks as follows: | ||
* ``` | ||
* test::{ | ||
* name:"Test #4", | ||
* type:"type", | ||
* body:{ | ||
* statement:"b", | ||
* session:{ | ||
* catalog:"b", | ||
* cwd:[ | ||
* "b" | ||
* ] | ||
* }, | ||
* status:SUCCESS, | ||
* expected:( | ||
* ROW | ||
* "b" | ||
* ( | ||
* ROW | ||
* "b" | ||
* INTEGER | ||
* ) | ||
* "c" | ||
* INTEGER | ||
* ) | ||
* } | ||
* } | ||
* ``` | ||
* @see org.partiql.planner.internal.typer.PlanTyperTestsPorted.TestCase.SuccessTestCase | ||
*/ | ||
class Success( | ||
name: String, | ||
expectedType: PType, | ||
statement: String, | ||
catalog: String, | ||
directory: List<String>, | ||
) : TyperTest(name, expectedType, statement, catalog, directory) { | ||
|
||
override fun assert() { | ||
val session = Session.builder() | ||
.catalog(catalog) | ||
.catalogs(*catalogs.toTypedArray()) | ||
.namespace(directory) | ||
.build() | ||
|
||
val collector = ProblemCollector() | ||
val ast = parser.parse(statement).root | ||
val plan = planner.plan(ast, session, collector).plan | ||
when (val statement = plan.statement) { | ||
is Statement.Query -> { | ||
assert(collector.problems.isEmpty()) { | ||
buildString { | ||
appendLine(collector.problems.toString()) | ||
appendLine() | ||
PlanPrinter.append(this, plan) | ||
} | ||
} | ||
val actual = statement.root.type | ||
assert(expectedType == actual) { | ||
buildString { | ||
appendLine() | ||
appendLine("Name: ${getName()}") | ||
appendLine("Expect: $expectedType") | ||
appendLine("Actual: $actual") | ||
appendLine("Statement: $statement") | ||
appendLine() | ||
PlanPrinter.append(this, plan) | ||
} | ||
} | ||
} | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* This represents an error test case. | ||
* | ||
* The serialization of an error test case looks as follows: | ||
* ``` | ||
* test::{ | ||
* name:"Pets should not be accessible #1", | ||
* type:"type", | ||
* body:{ | ||
* statement:"SELECT * FROM pets", | ||
* session:{ | ||
* catalog:"pql", | ||
* cwd:[ | ||
* ] | ||
* }, | ||
* status:FAILURE, | ||
* assertProblemExists:( | ||
* ERROR | ||
* '''Variable pets does not exist in the database environment and is not an attribute of the following in-scope variables [].''' | ||
* ) | ||
* } | ||
* } | ||
* ``` | ||
* The `assertProblemExists` is optional. | ||
* | ||
* @see org.partiql.planner.internal.typer.PlanTyperTestsPorted.TestCase.ErrorTestCase | ||
*/ | ||
class Failure( | ||
name: String, | ||
expectedType: PType?, | ||
private val assertion: TyperTestBuilder.ProblemAssertion?, | ||
statement: String, | ||
catalog: String, | ||
directory: List<String>, | ||
) : TyperTest(name, expectedType, statement, catalog, directory) { | ||
override fun assert() { | ||
val session = Session.builder() | ||
.catalog(catalog) | ||
.catalogs(*catalogs.toTypedArray()) | ||
.namespace(directory) | ||
.build() | ||
val collector = ProblemCollector() | ||
val ast = parser.parse(statement).root | ||
val plan = planner.plan(ast, session, collector).plan | ||
|
||
when (val statement = plan.statement) { | ||
is Statement.Query -> { | ||
assert(collector.problems.isNotEmpty()) { | ||
buildString { | ||
appendLine("Expected to find problems, but none were found.") | ||
appendLine() | ||
PlanPrinter.append(this, plan) | ||
} | ||
} | ||
if (expectedType != null) { | ||
assert(expectedType == statement.root.type) { | ||
buildString { | ||
appendLine() | ||
appendLine("Name: ${getName()}") | ||
appendLine("Expect: $expectedType") | ||
appendLine("Actual: ${statement.root.type}") | ||
appendLine("Statement: $statement") | ||
appendLine() | ||
PlanPrinter.append(this, plan) | ||
} | ||
} | ||
} | ||
assert(collector.problems.isNotEmpty()) { | ||
"Expected to find problems, but none were found." | ||
} | ||
if (assertion != null) { | ||
val problemFound = collector.problems.any { | ||
it.details.severity.name == assertion.severity && it.details.message == assertion.message | ||
} | ||
assert(problemFound) { | ||
"Could not find problem: ${assertion.severity}: ${assertion.message}." | ||
} | ||
} | ||
} | ||
} | ||
} | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Had some broader questions for porting the typing tests since I may be missing some context.
PlanTyperTestsPorted
(and related files)?PlanTyperTestsPorted
(and related files)?plan-typer-tests-ported.ion
file?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
test/partiql-test-interfaces
that can be imported as a test dependency across the project. Any particular subproject may have local serialized tests -- and once the team agrees upon the tests' correctness, they can be moved into the conformance test suite. If you'd like, I can move these interfaces immediately in this PR.PlanTyperTestsPorted
has to do with the serialization of errors, which is being worked on. I actually don't need to keep it right now. I can always resurface thePrintPlanTyperTestsPorted
from an older commit to re-serialize when the error serialization is addressed.tests
directory in any file you'd like.