Files
openai.openai-go/MIGRATION.md
2025-09-30 17:29:54 +00:00

285 lines
7.4 KiB
Markdown

# OpenAI Go Migration Guide
<a href="https://pkg.go.dev/github.com/openai/openai-go/v3"><img src="https://pkg.go.dev/badge/github.com/openai/openai-go/v3.svg" alt="Go Reference"></a>
This SDK includes breaking changes to improve the ergonomics of constructing parameters and accessing responses.
To reduce verbosity, the `openai.F(...)` and `param.Field[T]` have been removed.
All calls to `openai.F(...)` can be deleted.
The SDK now uses the <code>\`json:"...,omitzero"\`</code> struct tag to omit fields. Nested structs, arrays and maps
can be declared like normal.
The old SDK used interfaces for unions in requests, which required
a type assertion to access variants and fields. The new design uses
structs with a field for each variant, wherein only one field can be set.
These struct unions also expose 'Get' methods to access and mutate subfields
which may be shared by multiple variants.
# Request parameters
## Required primitives parameters serialize their zero values (`string`, `int64`, etc.)
> [!CAUTION]
>
> **This change can cause new behavior in existing code, without compiler warnings.**
While migrating, ensure that all required fields are explicitly set. A required primitive
field `Age` will use the <code>\`json:"age,required"\`</code> struct tag without `omitzero`.
If a required primitive field is not set, the zero value will be serialized.
This was not the case in with `param.Field[T]`.
```diff
type FooParams struct {
- Age param.Field[int64] `json:"age,required"`
- Name param.Field[string] `json:"name"`
+ Age int64 `json:"age,required"` // <== Notice no omitzero
+ Name param.Opt[string] `json:"name,omitzero"`
}
```
<table>
<tr>
<th>Previous</th>
<th>New</th>
</tr>
<tr>
<td>
```go
_ = FooParams{
Name: openai.String("Jerry")
}
`{"name": "Jerry"}` // (after serialization)
```
</td>
<td>
```go
_ = FooParams{
Name: openai.String("Jerry")
}
`{"name": "Jerry", "age": 0}` // <== Notice the age field
```
</td>
</tr>
</table>
The required field `"age"` is now present as `0`. Fields without the <code>\`json:"...,omitzero"\`</code> struct tag
are always serialized, including their zero values.
## Transition from `param.Field[T]` to `omitzero`
The `openai.F(...)` function and `param.Field[T]` type are no longer present in the new SDK.
To represent omitted fields, the SDK uses <a href="https://pkg.go.dev/encoding/json#Marshal"><code>\`json:"...,omitzero"\`</code> semantics</a> from Go 1.24+ for JSON encoding[^1]. `omitzero` always omits fields
with zero values.
In all cases other than optional primitives, `openai.F()` can simply be removed.
For optional primitive types, such as `param.Opt[string]`, you can use `openai.String(string)` to construct the value.
Similar functions exist for other primitive types like `openai.Int(int)`, `openai.Bool(bool)`, etc.
`omitzero` is used for fields whose type is either a struct, slice, map, string enum,
or wrapped optional primitive (e.g. `param.Opt[T]`). Required primitive fields don't use `omitzero`.
**Example User Code: Constructing a request**
```diff
foo = FooParams{
- RequiredString: openai.String("hello"),
+ RequiredString: "hello",
- OptionalString: openai.String("hi"),
+ OptionalString: openai.String("hi"),
- Array: openai.F([]BarParam{
- BarParam{Prop: ... }
- }),
+ Array: []BarParam{
+ BarParam{Prop: ... }
+ },
- RequiredObject: openai.F(BarParam{ ... }),
+ RequiredObject: BarParam{ ... },
- OptionalObject: openai.F(BarParam{ ... }),
+ OptionalObject: BarParam{ ... },
- StringEnum: openai.F[BazEnum]("baz-ok"),
+ StringEnum: "baz-ok",
}
```
**Internal SDK Code: Fields of a request struct:**
```diff
type FooParams struct {
- RequiredString param.Field[string] `json:"required_string,required"`
+ RequiredString string `json:"required_string,required"`
- OptionalString param.Field[string] `json:"optional_string"`
+ OptionalString param.Opt[string] `json:"optional_string,omitzero"`
- Array param.Field[[]BarParam] `json"array"`
+ Array []BarParam `json"array,omitzero"`
- Map param.Field[map[string]BarParam] `json"map"`
+ Map map[string]BarParam `json"map,omitzero"`
- RequiredObject param.Field[BarParam] `json:"required_object,required"`
+ RequiredObject BarParam `json:"required_object,omitzero,required"`
- OptionalObject param.Field[BarParam] `json:"optional_object"`
+ OptionalObject BarParam `json:"optional_object,omitzero"`
- StringEnum param.Field[BazEnum] `json:"string_enum"`
+ StringEnum BazEnum `json:"string_enum,omitzero"`
}
```
## Request Unions: Removing interfaces and moving to structs
For a type `AnimalUnionParam` which could be either a `CatParam | DogParam`.
<table>
<tr><th>Previous</th> <th>New</th></tr>
<tr>
<td>
```go
type AnimalParam interface {
ImplAnimalParam()
}
func (Dog) ImplAnimalParam() {}
func (Cat) ImplAnimalParam() {}
```
</td>
<td>
```go
type AnimalUnionParam struct {
OfCat *Cat `json:",omitzero,inline`
OfDog *Dog `json:",omitzero,inline`
}
```
</td>
</tr>
<tr style="background:rgb(209, 217, 224)">
<td>
```go
var dog AnimalParam = DogParam{
Name: "spot", ...
}
var cat AnimalParam = CatParam{
Name: "whiskers", ...
}
```
</td>
<td>
```go
dog := AnimalUnionParam{
OfDog: &DogParam{Name: "spot", ... },
}
cat := AnimalUnionParam{
OfCat: &CatParam{Name: "whiskers", ... },
}
```
</td>
</tr>
<tr>
<td>
```go
var name string
switch v := animal.(type) {
case Dog:
name = v.Name
case Cat:
name = v.Name
}
```
</td>
<td>
```go
// Accessing fields
var name *string = animal.GetName()
```
</td>
</tr>
</table>
## Sending explicit `null` values
The old SDK had a function `param.Null[T]()` which could set `param.Field[T]` to `null`.
The new SDK uses `param.Null[T]()` for to set a `param.Opt[T]` to `null`,
but `param.NullStruct[T]()` to set a param struct `T` to `null`.
```diff
- var nullPrimitive param.Field[int64] = param.Null[int64]()
+ var nullPrimitive param.Opt[int64] = param.Null[int64]()
- var nullStruct param.Field[BarParam] = param.Null[BarParam]()
+ var nullStruct BarParam = param.NullStruct[BarParam]()
```
## Sending custom values
The `openai.Raw[T](any)` function has been removed. All request structs now support a
`.WithExtraField(map[string]any)` method to customize the fields.
```diff
foo := FooParams{
A: param.String("hello"),
- B: param.Raw[string](12) // sending `12` instead of a string
}
+ foo.SetExtraFields(map[string]any{
+ "B": 12,
+ })
```
# Response Properties
## Checking for presence of optional fields
The `.IsNull()` method has been changed to `.Valid()` to better reflect its behavior.
```diff
- if !resp.Foo.JSON.Bar.IsNull() {
+ if resp.Foo.JSON.Bar.Valid() {
println("bar is present:", resp.Foo.Bar)
}
```
| Previous | New | Returns true for values |
| -------------- | ------------------------ | ----------------------- |
| `.IsNull()` | `!.Valid()` | `null` or Omitted |
| `.IsMissing()` | `.Raw() == resp.Omitted` | Omitted |
| | `.Raw() == resp.Null` |
## Checking Raw JSON of a response
The `.RawJSON()` method has moved to the parent of the `.JSON` property.
```diff
- resp.Foo.JSON.RawJSON()
+ resp.Foo.RawJSON()
```
[^1]: The SDK doesn't require Go 1.24, despite supporting the `omitzero` feature