Refine test

This commit is contained in:
Lennart Kats 2024-12-19 21:20:17 +01:00
parent 70f54dca12
commit 6d63b319d2
No known key found for this signature in database
GPG Key ID: 1EB8B57673197023
1 changed files with 62 additions and 80 deletions

View File

@ -589,11 +589,26 @@ func PresetsMock() *bundle.Bundle {
} }
} }
// Any fields that should be ignored in the completeness check
var PresetsIgnoredFields = map[string]string{ var PresetsIgnoredFields = map[string]string{
// Any fields that should be ignored in the completeness check "resources.pipelines.key.schema": "schema is still in private preview",
// Example: "resources.jobs.key.tasks[0].notebook_task.base_parameters": "catalog/schema are passed via job parameters",
// "resources.jobs.object.schema_something": "this property doesn't relate to the catalog/schema", "resources.jobs.key.tasks[0].python_wheel_task.named_parameters": "catalog/schema are passed via job parameters",
"resources.pipelines.key.schema": "schema is still in private preview", "resources.jobs.key.tasks[0].python_wheel_task.parameters": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].run_job_task.job_parameters": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].spark_jar_task.parameters": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].spark_python_task.parameters": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].spark_submit_task.parameters": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].sql_task.parameters": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].run_job_task.jar_params": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].run_job_task.notebook_params": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].run_job_task.pipeline_params": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].run_job_task.python_named_params": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].run_job_task.python_params": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].run_job_task.spark_submit_params": "catalog/schema are passed via job parameters",
"resources.jobs.key.tasks[0].run_job_task.sql_params": "catalog/schema are passed via job parameters",
"resources.pipelines.key.ingestion_definition.objects[0].schema": "schema name is under schema.source_schema/destination_schema",
"resources.schemas": "schema name of schemas is under resources.schemas.key.Name",
} }
func TestApplyPresetsCatalogSchema(t *testing.T) { func TestApplyPresetsCatalogSchema(t *testing.T) {
@ -627,6 +642,16 @@ func TestApplyPresetsCatalogSchema(t *testing.T) {
return root, err return root, err
})) }))
// Convert the recordedFields to a set for easier lookup
recordedSet := make(map[string]struct{})
for _, field := range recordedFields {
recordedSet[field.PathString] = struct{}{}
if i := strings.Index(field.PathString, "["); i >= 0 {
// For entries like resources.jobs.key.parameters[1].default, just add resources.jobs.key.parameters
recordedSet[field.PathString[:i]] = struct{}{}
}
}
// Stage 1: Apply presets before cleanup, should be no-op. // Stage 1: Apply presets before cleanup, should be no-op.
diags := bundle.Apply(ctx, b, mutator.ApplyPresets()) diags := bundle.Apply(ctx, b, mutator.ApplyPresets())
require.False(t, diags.HasError(), "unexpected error before cleanup: %v", diags.Error()) require.False(t, diags.HasError(), "unexpected error before cleanup: %v", diags.Error())
@ -663,7 +688,15 @@ func TestApplyPresetsCatalogSchema(t *testing.T) {
} }
// Stage 4: Check completeness // Stage 4: Check completeness
checkCompleteness(t, recordedFields) expectedFields := findCatalogSchemaFields()
assert.GreaterOrEqual(t, len(expectedFields), 42, "expected at least 42 catalog/schema fields, but got %d", len(expectedFields))
for _, field := range expectedFields {
if _, recorded := recordedSet[field]; !recorded {
if _, ignored := PresetsIgnoredFields[field]; !ignored {
t.Errorf("Field %s was not included in the catalog/schema presets test. If this is a new field, please add it to PresetsMock or PresetsIgnoredFields and add support for it as appropriate.", field)
}
}
}
} }
func verifyNoChangesBeforeCleanup(t *testing.T, rootVal dyn.Value, recordedFields []recordedField) { func verifyNoChangesBeforeCleanup(t *testing.T, rootVal dyn.Value, recordedFields []recordedField) {
@ -677,70 +710,34 @@ func verifyNoChangesBeforeCleanup(t *testing.T, rootVal dyn.Value, recordedField
} }
} }
func checkCompleteness(t *testing.T, recordedFields []recordedField) { // findCatalogSchemaFields finds all fields in config.Resources that might refer
t.Helper() // to a catalog or schema. Returns a slice of field paths.
func findCatalogSchemaFields() []string {
// Build a set for recorded fields
recordedSet := make(map[string]struct{})
for _, field := range recordedFields {
recordedSet[field.PathString] = struct{}{}
}
// Obtain the type of config.Resources
var r config.Resources
resourcesType := reflect.TypeOf(r)
// Track missing fields
var missingFields []string
// Keep track of visited types to prevent infinite loops (cycles)
visited := make(map[reflect.Type]struct{}) visited := make(map[reflect.Type]struct{})
var results []string
// Helper function to handle maps, slices, arrays, and nested pointers/interfaces // verifyTypeFields is a recursive function to verify the fields of a given type
verifyFieldType := func(fieldType reflect.Type, path string, fn func(reflect.Type, string)) { var walkTypeFields func(rt reflect.Type, path string)
switch fieldType.Kind() { walkTypeFields = func(rt reflect.Type, path string) {
case reflect.Slice, reflect.Array:
// For arrays/slices, inspect the element type
fn(fieldType.Elem(), path+"[0]")
case reflect.Map:
// For maps, inspect the value type
fn(fieldType.Elem(), path+".key")
case reflect.Ptr, reflect.Interface:
// For pointers/interfaces, inspect the element if it's a pointer
if fieldType.Kind() == reflect.Ptr {
fn(fieldType.Elem(), path)
}
case reflect.Struct:
// For structs, directly recurse into their fields
fn(fieldType, path)
default:
// For basic or unknown kinds, do nothing
}
}
// Recursive function to verify the fields of a given type.
var verifyTypeFields func(rt reflect.Type, path string)
verifyTypeFields = func(rt reflect.Type, path string) {
// Avoid cycles by skipping already visited types
if _, seen := visited[rt]; seen { if _, seen := visited[rt]; seen {
return return
} }
visited[rt] = struct{}{} visited[rt] = struct{}{}
switch rt.Kind() { switch rt.Kind() {
case reflect.Ptr, reflect.Interface: case reflect.Slice, reflect.Array:
// For pointers/interfaces, inspect the element type if available walkTypeFields(rt.Elem(), path+"[0]")
if rt.Kind() == reflect.Ptr { case reflect.Map:
verifyTypeFields(rt.Elem(), path) walkTypeFields(rt.Elem(), path+".key")
} case reflect.Ptr:
walkTypeFields(rt.Elem(), path)
case reflect.Struct: case reflect.Struct:
for i := 0; i < rt.NumField(); i++ { for i := 0; i < rt.NumField(); i++ {
ft := rt.Field(i) ft := rt.Field(i)
jsonTag := ft.Tag.Get("json") jsonTag := ft.Tag.Get("json")
if jsonTag == "" || jsonTag == "-" { if jsonTag == "" || jsonTag == "-" {
// Ignore field names when there's no JSON tag, // Ignore field names when there's no JSON tag, e.g. for Jobs.JobSettings
// e.g. for Jobs.JobSettings walkTypeFields(ft.Type, path)
verifyFieldType(ft.Type, path, verifyTypeFields)
continue continue
} }
@ -748,40 +745,25 @@ func checkCompleteness(t *testing.T, recordedFields []recordedField) {
fieldPath := path + "." + fieldName fieldPath := path + "." + fieldName
if isCatalogOrSchemaField(fieldName) { if isCatalogOrSchemaField(fieldName) {
// Only check if the field is a string results = append(results, fieldPath)
if ft.Type.Kind() == reflect.String {
if _, recorded := recordedSet[fieldPath]; !recorded {
if _, ignored := PresetsIgnoredFields[fieldPath]; !ignored {
missingFields = append(missingFields, fieldPath)
}
}
}
} }
verifyFieldType(ft.Type, fieldPath, verifyTypeFields) walkTypeFields(ft.Type, fieldPath)
} }
default:
// For other kinds at this level, do nothing
} }
} }
// Start from "resources" var r config.Resources
verifyTypeFields(resourcesType, "resources") walkTypeFields(reflect.TypeOf(r), "resources")
return results
// Report all missing fields
for _, field := range missingFields {
t.Errorf("Field %s was not included in the catalog/schema presets test. If this is a new field, please add it to PresetsMock or PresetsIgnoredFields and add support for it as appropriate.", field)
}
// Fail the test if there were any missing fields
if len(missingFields) > 0 {
t.FailNow()
}
} }
// isCatalogOrSchemaField returns true for a field names in config.Resources that we suspect could contain a catalog or schema name // isCatalogOrSchemaField returns true for a field names in config.Resources that we suspect could contain a catalog or schema name
func isCatalogOrSchemaField(name string) bool { func isCatalogOrSchemaField(name string) bool {
return strings.Contains(name, "catalog") || strings.Contains(name, "schema") return strings.Contains(name, "catalog") ||
strings.Contains(name, "schema") ||
strings.Contains(name, "parameters") ||
strings.Contains(name, "params")
} }
func removePlaceholders(value string) string { func removePlaceholders(value string) string {