Improve handling of JSON Schema in OpenAI API Response Context (#819)

* feat: add jsonschema.Validate and jsonschema.Unmarshal

* fix Sanity check

* remove slices.Contains

* fix Sanity check

* add SchemaWrapper

* update api_integration_test.go

* update method 'reflectSchema' to support 'omitempty' in JSON tag

* add GenerateSchemaForType

* update json_test.go

* update `Warp` to `Wrap`

* fix Sanity check

* fix Sanity check

* update api_internal_test.go

* update README.md

* update README.md

* remove jsonschema.SchemaWrapper

* remove jsonschema.SchemaWrapper

* fix Sanity check

* optimize code formatting
This commit is contained in:
eiixy
2024-08-25 01:06:08 +08:00
committed by GitHub
parent 5162adbbf9
commit a3bd2569ac
7 changed files with 412 additions and 30 deletions

136
jsonschema/validate_test.go Normal file
View File

@@ -0,0 +1,136 @@
package jsonschema_test
import (
"testing"
"github.com/sashabaranov/go-openai/jsonschema"
)
func Test_Validate(t *testing.T) {
type args struct {
data any
schema jsonschema.Definition
}
tests := []struct {
name string
args args
want bool
}{
// string integer number boolean
{"", args{data: "ABC", schema: jsonschema.Definition{Type: jsonschema.String}}, true},
{"", args{data: 123, schema: jsonschema.Definition{Type: jsonschema.String}}, false},
{"", args{data: 123, schema: jsonschema.Definition{Type: jsonschema.Integer}}, true},
{"", args{data: 123.4, schema: jsonschema.Definition{Type: jsonschema.Integer}}, false},
{"", args{data: "ABC", schema: jsonschema.Definition{Type: jsonschema.Number}}, false},
{"", args{data: 123, schema: jsonschema.Definition{Type: jsonschema.Number}}, true},
{"", args{data: false, schema: jsonschema.Definition{Type: jsonschema.Boolean}}, true},
{"", args{data: 123, schema: jsonschema.Definition{Type: jsonschema.Boolean}}, false},
{"", args{data: nil, schema: jsonschema.Definition{Type: jsonschema.Null}}, true},
{"", args{data: 0, schema: jsonschema.Definition{Type: jsonschema.Null}}, false},
// array
{"", args{data: []any{"a", "b", "c"}, schema: jsonschema.Definition{
Type: jsonschema.Array, Items: &jsonschema.Definition{Type: jsonschema.String}},
}, true},
{"", args{data: []any{1, 2, 3}, schema: jsonschema.Definition{
Type: jsonschema.Array, Items: &jsonschema.Definition{Type: jsonschema.String}},
}, false},
{"", args{data: []any{1, 2, 3}, schema: jsonschema.Definition{
Type: jsonschema.Array, Items: &jsonschema.Definition{Type: jsonschema.Integer}},
}, true},
{"", args{data: []any{1, 2, 3.4}, schema: jsonschema.Definition{
Type: jsonschema.Array, Items: &jsonschema.Definition{Type: jsonschema.Integer}},
}, false},
// object
{"", args{data: map[string]any{
"string": "abc",
"integer": 123,
"number": 123.4,
"boolean": false,
"array": []any{1, 2, 3},
}, schema: jsonschema.Definition{Type: jsonschema.Object, Properties: map[string]jsonschema.Definition{
"string": {Type: jsonschema.String},
"integer": {Type: jsonschema.Integer},
"number": {Type: jsonschema.Number},
"boolean": {Type: jsonschema.Boolean},
"array": {Type: jsonschema.Array, Items: &jsonschema.Definition{Type: jsonschema.Number}},
},
Required: []string{"string"},
}}, true},
{"", args{data: map[string]any{
"integer": 123,
"number": 123.4,
"boolean": false,
"array": []any{1, 2, 3},
}, schema: jsonschema.Definition{Type: jsonschema.Object, Properties: map[string]jsonschema.Definition{
"string": {Type: jsonschema.String},
"integer": {Type: jsonschema.Integer},
"number": {Type: jsonschema.Number},
"boolean": {Type: jsonschema.Boolean},
"array": {Type: jsonschema.Array, Items: &jsonschema.Definition{Type: jsonschema.Number}},
},
Required: []string{"string"},
}}, false},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := jsonschema.Validate(tt.args.schema, tt.args.data); got != tt.want {
t.Errorf("Validate() = %v, want %v", got, tt.want)
}
})
}
}
func TestUnmarshal(t *testing.T) {
type args struct {
schema jsonschema.Definition
content []byte
v any
}
var result1 struct {
String string `json:"string"`
Number float64 `json:"number"`
}
var result2 struct {
String string `json:"string"`
Number float64 `json:"number"`
}
tests := []struct {
name string
args args
wantErr bool
}{
{"", args{
schema: jsonschema.Definition{
Type: jsonschema.Object,
Properties: map[string]jsonschema.Definition{
"string": {Type: jsonschema.String},
"number": {Type: jsonschema.Number},
},
},
content: []byte(`{"string":"abc","number":123.4}`),
v: &result1,
}, false},
{"", args{
schema: jsonschema.Definition{
Type: jsonschema.Object,
Properties: map[string]jsonschema.Definition{
"string": {Type: jsonschema.String},
"number": {Type: jsonschema.Number},
},
Required: []string{"string", "number"},
},
content: []byte(`{"string":"abc"}`),
v: result2,
}, true},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
err := jsonschema.VerifySchemaAndUnmarshal(tt.args.schema, tt.args.content, tt.args.v)
if (err != nil) != tt.wantErr {
t.Errorf("Unmarshal() error = %v, wantErr %v", err, tt.wantErr)
} else if err == nil {
t.Logf("Unmarshal() v = %+v\n", tt.args.v)
}
})
}
}