/* Copyright The Helm Authors. 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 http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ package util import ( "net/http" "net/http/httptest" "os" "strings" "testing" chart "helm.sh/helm/v4/pkg/chart/v2" ) func TestValidateAgainstSingleSchema(t *testing.T) { values, err := ReadValuesFile("./testdata/test-values.yaml") if err != nil { t.Fatalf("Error reading YAML file: %s", err) } schema, err := os.ReadFile("./testdata/test-values.schema.json") if err != nil { t.Fatalf("Error reading YAML file: %s", err) } if err := ValidateAgainstSingleSchema(values, schema); err != nil { t.Errorf("Error validating Values against Schema: %s", err) } } func TestValidateAgainstInvalidSingleSchema(t *testing.T) { values, err := ReadValuesFile("./testdata/test-values.yaml") if err != nil { t.Fatalf("Error reading YAML file: %s", err) } schema, err := os.ReadFile("./testdata/test-values-invalid.schema.json") if err != nil { t.Fatalf("Error reading YAML file: %s", err) } var errString string if err := ValidateAgainstSingleSchema(values, schema); err == nil { t.Fatalf("Expected an error, but got nil") } else { errString = err.Error() } expectedErrString := `"file:///values.schema.json#" is not valid against metaschema: jsonschema validation failed with 'https://json-schema.org/draft/2020-12/schema#' - at '': got number, want boolean or object` if errString != expectedErrString { t.Errorf("Error string :\n`%s`\ndoes not match expected\n`%s`", errString, expectedErrString) } } func TestValidateAgainstSingleSchemaNegative(t *testing.T) { values, err := ReadValuesFile("./testdata/test-values-negative.yaml") if err != nil { t.Fatalf("Error reading YAML file: %s", err) } schema, err := os.ReadFile("./testdata/test-values.schema.json") if err != nil { t.Fatalf("Error reading JSON file: %s", err) } var errString string if err := ValidateAgainstSingleSchema(values, schema); err == nil { t.Fatalf("Expected an error, but got nil") } else { errString = err.Error() } expectedErrString := `- at '': missing property 'employmentInfo' - at '/age': minimum: got -5, want 0 ` if errString != expectedErrString { t.Errorf("Error string :\n`%s`\ndoes not match expected\n`%s`", errString, expectedErrString) } } const subchartSchema = `{ "$schema": "http://json-schema.org/draft-07/schema#", "title": "Values", "type": "object", "properties": { "age": { "description": "Age", "minimum": 0, "type": "integer" } }, "required": [ "age" ] } ` const subchartSchema2020 = `{ "$schema": "https://json-schema.org/draft/2020-12/schema", "title": "Values", "type": "object", "properties": { "data": { "type": "array", "contains": { "type": "string" }, "unevaluatedItems": { "type": "number" } } }, "required": ["data"] } ` func TestValidateAgainstSchema(t *testing.T) { subchartJSON := []byte(subchartSchema) subchart := &chart.Chart{ Metadata: &chart.Metadata{ Name: "subchart", }, Schema: subchartJSON, } chrt := &chart.Chart{ Metadata: &chart.Metadata{ Name: "chrt", }, } chrt.AddDependency(subchart) vals := map[string]interface{}{ "name": "John", "subchart": map[string]interface{}{ "age": 25, }, } if err := ValidateAgainstSchema(chrt, vals); err != nil { t.Errorf("Error validating Values against Schema: %s", err) } } func TestValidateAgainstSchemaNegative(t *testing.T) { subchartJSON := []byte(subchartSchema) subchart := &chart.Chart{ Metadata: &chart.Metadata{ Name: "subchart", }, Schema: subchartJSON, } chrt := &chart.Chart{ Metadata: &chart.Metadata{ Name: "chrt", }, } chrt.AddDependency(subchart) vals := map[string]interface{}{ "name": "John", "subchart": map[string]interface{}{}, } var errString string if err := ValidateAgainstSchema(chrt, vals); err == nil { t.Fatalf("Expected an error, but got nil") } else { errString = err.Error() } expectedErrString := `subchart: - at '': missing property 'age' ` if errString != expectedErrString { t.Errorf("Error string :\n`%s`\ndoes not match expected\n`%s`", errString, expectedErrString) } } func TestValidateAgainstSchema2020(t *testing.T) { subchartJSON := []byte(subchartSchema2020) subchart := &chart.Chart{ Metadata: &chart.Metadata{ Name: "subchart", }, Schema: subchartJSON, } chrt := &chart.Chart{ Metadata: &chart.Metadata{ Name: "chrt", }, } chrt.AddDependency(subchart) vals := map[string]interface{}{ "name": "John", "subchart": map[string]interface{}{ "data": []any{"hello", 12}, }, } if err := ValidateAgainstSchema(chrt, vals); err != nil { t.Errorf("Error validating Values against Schema: %s", err) } } func TestValidateAgainstSchema2020Negative(t *testing.T) { subchartJSON := []byte(subchartSchema2020) subchart := &chart.Chart{ Metadata: &chart.Metadata{ Name: "subchart", }, Schema: subchartJSON, } chrt := &chart.Chart{ Metadata: &chart.Metadata{ Name: "chrt", }, } chrt.AddDependency(subchart) vals := map[string]interface{}{ "name": "John", "subchart": map[string]interface{}{ "data": []any{12}, }, } var errString string if err := ValidateAgainstSchema(chrt, vals); err == nil { t.Fatalf("Expected an error, but got nil") } else { errString = err.Error() } expectedErrString := `subchart: - at '/data': no items match contains schema - at '/data/0': got number, want string ` if errString != expectedErrString { t.Errorf("Error string :\n`%s`\ndoes not match expected\n`%s`", errString, expectedErrString) } } func TestHTTPURLLoader_Load(t *testing.T) { // Test successful JSON schema loading t.Run("successful load", func(t *testing.T) { server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, _ *http.Request) { w.Header().Set("Content-Type", "application/json") w.WriteHeader(http.StatusOK) w.Write([]byte(`{"type": "object", "properties": {"name": {"type": "string"}}}`)) })) defer server.Close() loader := newHTTPURLLoader() result, err := loader.Load(server.URL) if err != nil { t.Fatalf("Expected no error, got: %v", err) } if result == nil { t.Fatal("Expected result to be non-nil") } }) t.Run("HTTP error status", func(t *testing.T) { server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, _ *http.Request) { w.WriteHeader(http.StatusNotFound) })) defer server.Close() loader := newHTTPURLLoader() _, err := loader.Load(server.URL) if err == nil { t.Fatal("Expected error for HTTP 404") } if !strings.Contains(err.Error(), "404") { t.Errorf("Expected error message to contain '404', got: %v", err) } }) }