Skip to content

Commit

Permalink
fix: function call error due to nil properties (429) (sashabaranov#431)
Browse files Browse the repository at this point in the history
* fix: fix function call error due to nil properties (429)

* refactor: refactoring initializeProperties func in jsonschema pkg (429)
  • Loading branch information
vvatanabe authored Jul 9, 2023
1 parent 181fc2a commit f028c28
Show file tree
Hide file tree
Showing 2 changed files with 225 additions and 1 deletion.
25 changes: 24 additions & 1 deletion jsonschema/json.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@
// and/or pass in the schema in []byte format.
package jsonschema

import "encoding/json"

type DataType string

const (
Expand All @@ -17,7 +19,7 @@ const (
)

// Definition is a struct for describing a JSON Schema.
// It is fairly limited and you may have better luck using a third-party library.
// It is fairly limited, and you may have better luck using a third-party library.
type Definition struct {
// Type specifies the data type of the schema.
Type DataType `json:"type,omitempty"`
Expand All @@ -33,3 +35,24 @@ type Definition struct {
// Items specifies which data type an array contains, if the schema type is Array.
Items *Definition `json:"items,omitempty"`
}

func (d *Definition) MarshalJSON() ([]byte, error) {
d.initializeProperties()
return json.Marshal(*d)
}

func (d *Definition) initializeProperties() {
if d.Properties == nil {
d.Properties = make(map[string]Definition)
return
}

for k, v := range d.Properties {
if v.Properties == nil {
v.Properties = make(map[string]Definition)
} else {
v.initializeProperties()
}
d.Properties[k] = v
}
}
201 changes: 201 additions & 0 deletions jsonschema/json_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,201 @@
package jsonschema_test

import (
"encoding/json"
"reflect"
"testing"

. "github.com/sashabaranov/go-openai/jsonschema"
)

func TestDefinition_MarshalJSON(t *testing.T) {
tests := []struct {
name string
def Definition
want string
}{
{
name: "Test with empty Definition",
def: Definition{},
want: `{"properties":{}}`,
},
{
name: "Test with Definition properties set",
def: Definition{
Type: String,
Description: "A string type",
Properties: map[string]Definition{
"name": {
Type: String,
},
},
},
want: `{
"type":"string",
"description":"A string type",
"properties":{
"name":{
"type":"string",
"properties":{}
}
}
}`,
},
{
name: "Test with nested Definition properties",
def: Definition{
Type: Object,
Properties: map[string]Definition{
"user": {
Type: Object,
Properties: map[string]Definition{
"name": {
Type: String,
},
"age": {
Type: Integer,
},
},
},
},
},
want: `{
"type":"object",
"properties":{
"user":{
"type":"object",
"properties":{
"name":{
"type":"string",
"properties":{}
},
"age":{
"type":"integer",
"properties":{}
}
}
}
}
}`,
},
{
name: "Test with complex nested Definition",
def: Definition{
Type: Object,
Properties: map[string]Definition{
"user": {
Type: Object,
Properties: map[string]Definition{
"name": {
Type: String,
},
"age": {
Type: Integer,
},
"address": {
Type: Object,
Properties: map[string]Definition{
"city": {
Type: String,
},
"country": {
Type: String,
},
},
},
},
},
},
},
want: `{
"type":"object",
"properties":{
"user":{
"type":"object",
"properties":{
"name":{
"type":"string",
"properties":{}
},
"age":{
"type":"integer",
"properties":{}
},
"address":{
"type":"object",
"properties":{
"city":{
"type":"string",
"properties":{}
},
"country":{
"type":"string",
"properties":{}
}
}
}
}
}
}
}`,
},
{
name: "Test with Array type Definition",
def: Definition{
Type: Array,
Items: &Definition{
Type: String,
},
Properties: map[string]Definition{
"name": {
Type: String,
},
},
},
want: `{
"type":"array",
"items":{
"type":"string",
"properties":{
}
},
"properties":{
"name":{
"type":"string",
"properties":{}
}
}
}`,
},
}

for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
gotBytes, err := json.Marshal(&tt.def)
if err != nil {
t.Errorf("Failed to Marshal JSON: error = %v", err)
return
}

var got map[string]interface{}
err = json.Unmarshal(gotBytes, &got)
if err != nil {
t.Errorf("Failed to Unmarshal JSON: error = %v", err)
return
}

wantBytes := []byte(tt.want)
var want map[string]interface{}
err = json.Unmarshal(wantBytes, &want)
if err != nil {
t.Errorf("Failed to Unmarshal JSON: error = %v", err)
return
}

if !reflect.DeepEqual(got, want) {
t.Errorf("MarshalJSON() got = %v, want %v", got, want)
}
})
}
}

0 comments on commit f028c28

Please sign in to comment.