Releases: danielgtaylor/huma
v2.20.0
Overview
Sponsors
Thank you to Zuplo for sponsoring the project.
Fixed Validation Tags & Referenced Schemas
Sometimes schemas and schemas in references wound up getting overwritten with default values for validation like minimum
, maximum
, minLength
, maxLength
, as well as skipping calling TransformSchema
. This is now fixed to behave as expected by derefencing structs and only setting validation values when the tag is actually present.
Customizable Validation Errors
Validation errors are now completely customizable as globals in the library.
// Set a custom message for minimum validation failure.
huma.MsgExpectedMinimumNumber = "expected number bigger than or equal to %v"
You can also change the function used to format error messages via huma.ErrorFormatter
. This must be overwritten before generating any schemas, as the generated messages are cached at schema creation time.
var ErrorFormatter func(format string, a ...any) string = fmt.Sprintf
Nullable Slices
Due to the way nil
slices get encoded in Go as null
in JSON and other formats, we now generate array schemas as nullable by default. See also the discussion and links to JSON v2 at golang/go#37711.
What's Changed
- fix(schema): proper minLength, maxLength in generated schemas. by @superstas in #513
- fix(schema): proper minimum, maximum in generated schemas. by @superstas in #515
- fix(schema): add dereferencing for SchemaFromType by @superstas in #518
- feat: add zuplo sponsorship by @danielgtaylor in #501
- docs: fix formatting in sponsors by @danielgtaylor in #523
- feat: make validation error messages customizable by @smacker in #520
- fix: nullable schemas for arrays/slices by @lsdch in #527
New Contributors
- @superstas made their first contribution in #513
- @smacker made their first contribution in #520
Full Changelog: v2.19.0...v2.20.0
v2.19.0
Overview
Sponsors
A big thank you to our new sponsor:
Multipart Form File Metadata
It's now possible to get filename & size metadata information from multipart form files.
huma.Post(api, "/form-example", func(ctx context.Context, input *struct{
RawBody huma.MultipartFormFiles[struct {
HelloWorld huma.FormFile `form:"file" contentType:"text/plain" required:"true"`
}]
}) (*struct{}, error) {
fileData := input.RawBody.Data()
fmt.Println( fileData.HelloWorld.Filename)
fmt.Println( fileData.HelloWorld.Size)
}
Easier Custom Context When Testing
It's now easier to pass a custom context to operations in the test API. Instead of having to create a custom request with its own context and manually call the adapter you can now use the methods like GetCtx
instead of Get
.
_, api := humatest.New(t)
ctx := context.Background() // define your necessary context
resp := api.GetCtx(ctx, "/greeting/world") // provide it using the 'Ctx' suffixed methods
Exploded Query Params
It's now possible to use the OpenAPI explode
feature where query params are passed multiple times rather than using comma separated values.
huma.Get(api, "/example", func(ctx context.Context, input *struct{
Value []string `query:"value,explode"`
}) (*struct{}, error) {
fmt.Println(input.Value)
return nil, nil
})
You can then make requests like GET /example?value=foo&value=bar
.
Autopatch Schema Improvements
Autopatch now uses the PUT
schema (modified to all be optional) rather than just relying on an object
with any allowed properties, which improves documentation for users. This is automatic so there is no need to configure anything new.
Other Improvements
- Performance improvement by removing the response body from panics which could be very large.
- Fixes to min/max items schema generation when using arrays.
- Remove
slices
dependency to continue to support Go 1.20 until 1.23 is released (we will support the latest two major versions just like the Go project itself)
What's Changed
- fix: expose metadata of decoded multipart form files by @lsdch in #466
- Extend TestAPI interface to allow for custom context.Context by @coury-clark in #469
- Update README_CN.md by @Ivlyth in #477
- remove response body from panic message by @austincollinpena in #479
- fix(schema): proper array minItems, maxItems and doc reporting in generated schemas by @Grumpfy in #485
- fix: remove slices dependency to better support Go 1.20 by @danielgtaylor in #497
- Implement exploded query parameters by @csmarchbanks in #498
- Improvement Autopatch (adding a body) by @ScriptType in #496
- docs: add new sponsor by @danielgtaylor in #506
New Contributors
- @coury-clark made their first contribution in #469
- @Ivlyth made their first contribution in #477
- @Grumpfy made their first contribution in #485
- @csmarchbanks made their first contribution in #498
- @ScriptType made their first contribution in #496
Full Changelog: v2.18.0...v2.19.0
v2.18.0
Overview
Better Multipart Files Uploads
You can now use huma.MultipartFormFiles[YourType]
as the request's RawBody
in order to handle multipart file uploads. Files can be limited to specific content types and required or optional. Example:
type FileData struct {
// This is an example, any number of `multipart.File` fields can be defined.
// Nested structs are not supported.
SomeFile multipart.File `form-data:"some_file" content-type:"image/png" required:"true"`
SeveralFiles []multipart.File `form-data:"several_files" content-type:"image/png,image/jpeg" required:"true"`
}
type FileHandlerInput struct {
RawBody huma.MultipartFormFiles[FileData]
}
func FileHandler(ctx context.Context, input *FileHandlerInput) (*struct{}, error) {
fileData := input.RawBody.Data()
DoSomeThingWith(fileData.SomeFile)
OrSomethingElseWith(fileData.SeveralFiles)
}
huma.Register(api,
huma.Operation{
Path: "/handle-files",
Method: http.MethodPost,
OperationID: "Handle files",
}, FileHandler)
Schema Transformers
It's now possible to have types implement a schema transformer, which lets them modify the generated schema for the type. This option lives in between using the generated types and providing your own schema, and makes it a bit easier to modify the generated schema by not needing you to call into the registry manually. This is the interface:
type SchemaTransformer interface {
TransformSchema(r Registry, s *Schema) *Schema
}
Simple example:
type MyInput struct {
Field string `json:"field"`
}
func (i *MyInput) TransformSchema(r huma.Registry, s *huma.Schema) *huma.Schema {
s.Description = "I am an override"
return s
}
What's Changed
- Fix typo in SSE doc by @op-tmplt in #448
- Feature: handling files from
multipart/form-data
request by @lsdch in #415 - feat: allow modifying generated schema by implementing SchemaTransformer interface by @lsdch in #456
- fix: type schema description not work by @fourcels in #462
New Contributors
Full Changelog: v2.17.0...v2.18.0
v2.17.0
Overview
Convenience Method Operation Modifiers
You can now add modifier functions when registering operations using convenience methods like huma.Get
:
func OperationTags(tags ...string) func(o *Operation) {
return func(o *Operation) {
o.Tags = tags
}
}
huma.Get(api, "/demo", myHandler, OperationTags("one", "two"))
Use this to build out whatever functionality you need to simplify registration. These can also be composed & joined easily to give a concise way to register operations without falling back to huma.Register
directly.
Request Remote Address Access
It's now possible to access the requests's RemoteAddr
field through huma.Context
for use in middleware and/or resolvers, which works with all routers:
func MyMiddleware(ctx huma.Context, next func(huma.Context)) {
fmt.Println("Request address", ctx.RemoteAddr())
next(ctx)
}
Fix Unmarshaling of Embedded Body
Unmarshaling of embedded structs with a body field now works as expected:
type BodyContainer struct {
Body struct {
Name string `json:"name"`
}
}
huma.Register(api, huma.Operation{
Method: http.MethodPost,
Path: "/body",
}, func(ctx context.Context, input *struct {
BodyContainer
}) (*struct{}, error) {
// Do something with `input.Body.Name`...
return nil, nil
})
What's Changed
- feat: convenience methods with operationHandlers by @fourcels in #434
- fix: race condition in CLI tests by @danielgtaylor in #436
- docs: add new sponsor and quote by @danielgtaylor in #437
- fix: unmarshaling of request body when using struct embedding by @lsdch in #440
- feat: add request remote address to context by @danielgtaylor in #447
New Contributors
Full Changelog: v2.16.0...v2.17.0
v2.16.0
Overview
Drop Deprecated Chi v4 Adapter
This release drops Chi v4 support. Chi v4 has been deprecated & replaced with Chi v5 since 2021, and was included to help migrate from Huma v1 to v2. Huma v2 has now been out for about six months. Keeping Chi v4 in the main package results in extra dependencies and deprecation warnings from automated tools like dependabot. It is highly recommended to discontinue using Chi v4.
If you wish to continue using Chi v4, please copy the v4 adapter from https://github.com/danielgtaylor/huma/blob/v2.15.0/adapters/humachi/humachi.go into your own project. With that simple change you can migrate off at your own pace while continuing to get new Huma releases.
Better Panics
Some panic messages have been improved to be more helpful for users, for example letting them know that the response type must be a struct rather than some confusing reflection panic.
What's Changed
- feat!: drop chi v4 adapter by @costela in #427
- panic with more descriptive messages by @ssoroka in #431
Full Changelog: v2.15.0...v2.16.0
v2.15.0
Sponsors
A big thank you to our new sponsor:
Overview
Schema Discriminator Support
Basic support for the OpenAPI 3.1 discriminator which gives hints to clients that the value of a field defines the type of the response. This is not currently exposed via tags but can be used when manually creating schemas:
s := &huma.Schema{
OneOf: []*huma.Schema{
{Type: "object", Properties: map[string]*huma.Schema{
"type": {Type: "string", Enum: []any{"foo"}},
"foo": {Type: "string"},
}},
{Type: "object", Properties: map[string]*huma.Schema{
"type": {Type: "string", Enum: []any{"bar"}},
"bar": {Type: "string"},
}},
},
Discriminator: &huma.Discriminator{
PropertyName: "type",
Mapping: map[string]string{
"foo": "#/components/schemas/Foo",
"bar": "#/components/schemas/Bar",
},
},
}
Anonymous Struct Name Hints
Allow providing name hints via field tags for anonymous structs defined inline. This gives a bit more control over the JSON Schema type names:
type EndpointInput struct {
Body struct {
SomeData string `json:"some_data"`
} `name-hint:"SomeName"`
}
Better File Upload UI Support
A contentMediaType
field is generated for fields which are format: "binary"
which enables a better UI for uploading files in the generated docs.
Bug Fixes
The generated $schema
field now uses http
instead of https
when the host is 127.0.0.1
. Previously this was only the case for localhost
.
Pointer types with custom schemas are now better supported by dereferencing the pointer to the underlying type before checking for the custom schema interface implementation.
The built-in Flow router has a fix applied to handle path params that are percent encoded with slashes. Fix has also been submitted upstream.
Fixed a possible panic in the schema link transformer when passed nil
body types.
Updated the precondition error locations to match the rest of the project. request.headers.If-Match
→ headers.If-Match
as we no longer explicitly state it's in the request. It's always in the request.
Fixed an example in the docs that was resulting in an error due to a typo.
What's Changed
- Use plain http in $schema for 127.0.0.1 by @davidolrik in #408
- fix: check if struct implements interface SchemaProvider after deref by @lennycampino in #413
- fix: Add support for Percent-Encoding RFC 3986 by @chilic in #411
- fix: more robust nil check for link transformer by @danielgtaylor in #418
- feat: basic schema discriminator support by @danielgtaylor in #419
- fix: precondition error location to match other errors by @danielgtaylor in #420
- Allow providing name hints using tags for anonymous body structs by @lsdch in #416
- fix: cli name example in docs by @danielgtaylor in #421
- Add contentMediaType field to generated schema when format is binary by @lsdch in #422
New Contributors
- @davidolrik made their first contribution in #408
- @lennycampino made their first contribution in #413
- @chilic made their first contribution in #411
Full Changelog: v2.14.0...v2.15.0
v2.14.0
Overview
Support for netip.Addr
Type
In addition to a net.IP
, request inputs can now use a netip.Addr
type, passing in a value like 127.0.0.1
from the client in e.g. query params and it will parse into the correct type with validation.
Error Wrapping Improvements
Arbitrary headers can now be added if needed by wrapping errors. For example:
return nil, huma.ErrorWithHeaders(
huma.Error404NotFound("thing not found"),
http.Header{
"Cache-Control": {"no-store"},
},
)
This uses Go's built-in error wrapping functionality and works via errors.Is
/errors.As
for any error which satisfies the huma.HeadersError
interface.
What's Changed
- test: more tests for
cookie
by @behnambm in #391 - build(deps): bump golang.org/x/net from 0.21.0 to 0.23.0 by @dependabot in #394
- Support
netip.Addr
as a special type by @wolveix in #396 - feat: wrap errors to set headers by @danielgtaylor in #387
New Contributors
Full Changelog: v2.13.1...v2.14.0
v2.13.1
Overview
This patch release fixes a small bug where a 204 No Content or 304 Not Modified which included a body would result in a panic. After this fix the body is ignored, making it easier to use the huma.Status304NotModified()
utility.
What's Changed
- feature: add Chinese document by @gg1229505432 in #385
- fix: do no try to write 204/304 body if present by @danielgtaylor in #390
New Contributors
- @gg1229505432 made their first contribution in #385
Full Changelog: v2.13.0...v2.13.1
v2.13.0
Overview
This release includes some minor but important changes that may break some existing users by removing deprecated functionality. I'm really sorry for the potential breaks and will do my best to adhere more strictly to semantic versioning in the future. Be aware that these changes are not taken lightly and possible alternatives were considered. I have also tried to group all the small breaks/changes into a single release to mitigate repeated painful upgrades. Building this type of library is hard and I feel for your frustrations and appreciate your understanding!
As of this release, you can now build a full-fledged Huma service with zero additional hard dependencies (if using the Go 1.22+ built-in huma.ServeMux
router and only JSON).
Removal of deprecated huma.NewCLI
Back in version 2.8.0 the huma.NewCLI
functionality was deprecated in favor of humacli.New
. The deprecated call is now removed.
Caution
This has the potential to break existing users, but is being treated as a bug fix to remove a hard dependency on spf13/cobra
. If you are already using humacli.New
there is no need to change anything.
Make CBOR support optional
CBOR is now made optional in the default config. If you wish to continue using it, you must opt-in to a new import, which automatically registers the format with the default configuration:
import (
"github.com/danielgtaylor/huma/v2"
_ "github.com/danielgtaylor/huma/v2/formats/cbor"
)
This new behavior is documented at https://huma.rocks/features/response-serialization/#default-formats. In the future this also makes it easier to support other additional formats without adding hard dependencies.
Warning
While not a breaking change per se, without adding the new import your clients will no longer be able to request CBOR and will get JSON responses instead.
humatest
no longer requires Chi
The humatest
package now uses a tiny internal router rather than relying on the Chi router, which introduced extra dependencies especially for people not using Chi as their main router package. You can continue to use humatest.Wrap
to wrap any router you like.
Also new are humatest.DumpRequest
/humatest.DumpResponse
and humatest.PrintRequest
/humatest.PrintResponse
utility functions which function similar to httptest.DumpRequest
/httptest.DumpResponse
but pretty print the body JSON to make debugging and showing examples easier. Usage example: https://go.dev/play/p/QQ9Bi7iws12.
Caution
This is technically a small breaking change, but is being treated as a bug fix to remove the dependency and make the package more future-proof by not relying on any single external router package and returning http.Handler
for the router instead.
Per-operation Middleware
You can now attach middleware to a huma.Operation
during registration. These middlewares will only run for that specific operation rather than for all operations:
huma.Register(api, huma.Operation{
Method: http.MethodGet,
Path: "/demo",
Middlewares: huma.Middlewares{
func(ctx huma.Context, next func(huma.Context)) {
// TODO: implement middleware here...
next(ctx)
},
},
}, func(ctx context.Context, input *struct{}) (*struct{}, error) {
// TODO: implement handler here...
return nil, nil
})
See also https://huma.rocks/features/middleware/#operations.
Add ctx.Status()
You can now get the response status from a huma.Context
in router-agnostic middleware, enabling easier logging/metrics/traces of the response status code.
func MyMiddleware(ctx huma.Context, next func(huma.Context)) {
next(ctx)
fmt.Println("Response status is", ctx.Status())
}
Better parameter descriptions
Some tools, notably SwaggerUI, require parameters to have descriptions to display properly (descriptions in the schema are not sufficient). These are now generated by Huma, so users opting to render docs with SwaggerUI should now get nicer output.
Bug Fixes
- The
humafiber
adapter accepts the correct group interface now. - Schema name text transformations for generics are now applied to hints as well as generated names
- Do not crash when
reflect.StructOf
fails in schema link transformer - Use
errors.As
when looking forhuma.StatusError
to enable error wrapping - Better error messages when missing OpenAPI config
What's Changed
- fix: NewWithGroup accepting fiber.Router by @barklan in #361
- fix: remove deprecated cli functionality to limit dependencies by @danielgtaylor in #363
- fix: apply schema name text transforms to type hints by @danielgtaylor in #367
- fix: vendor casing library by @danielgtaylor in #368
- fix: do not require Chi for humatest by @danielgtaylor in #370
- Add the ability to get ctx.Status() by @ssoroka-tc in #369
- feat: add param descriptions from schemas by @danielgtaylor in #374
- fix: catch reflect.StructOf panics in schema transformer by @danielgtaylor in #375
- fix: use errors.As to get StatusError, enabling wrapping by @danielgtaylor in #381
- feat: add per-operation router-agnostic middleware by @danielgtaylor in #382
- must specify openapi config by @ssoroka in #383
- fix: make cbor support optional by @danielgtaylor in #372
New Contributors
- @barklan made their first contribution in #361
- @ssoroka-tc made their first contribution in #369
- @ssoroka made their first contribution in #383
Full Changelog: v2.12.0...v2.13.0
v2.12.0
Overview
Support json.RawMessage
There is now better support in the validator for json.RawMessage
, which turns into an empty schema (allow anything).
type Demo struct {
Field1 string `json:"field1"`
Field2 json.RawMessage `json:"field2"`
}
OpenAPI 3.0.3
It's now easy to get OpenAPI 3.0.3 from a Huma service for tools that aren't completely compatible with OpenAPI 3.1 just yet. See https://huma.rocks/features/openapi-generation/. The new specs are available at /openapi-3.0.json
and /openapi-3.0.yaml
by default. Programmatic access is available via api.OpenAPI().Downgrade()
.
Better Support for Optional / Nullable Types
Huma now has better support for optional/nullable schemas. Broadly speaking, omitempty
controls whether a field is optional/required and the use of a pointer controls nullability. As described in:
- https://huma.rocks/features/request-validation/#optional-required
- https://huma.rocks/features/request-validation/#nullable
Huma generates type arrays from pointers to simple scalar types (boolean
, integer
, number
, string
) like "type": ["string", "null"]
. This is especially useful for frontend languages like Javascript/Typescript where there is a distinction between explicit null vs. undefined and an SDK generator might produce a type union like:
// Generated Typescript example
type GreetingOutputBody = {
message: string | null;
};
In addition to the default automatic behavior for determining optional/nullable, these can be manually overridden by the user via the required:"true|false"
and nullable:"true|false"
field tags, ensuring you have full control over how the schema is generated.
Important
Types which result in a JSON array
or object
are not marked as nullable by default, due to the complexity of modeling this in JSON Schema and poor support from SDK generators for Go and some other languages. This may change in a future release. You can mark a struct as nullable manually using a _
field with a nullable:"true"
tag as described in the docs linked above.
What's Changed
- docs: add sponsors to docs homepage by @danielgtaylor in #349
- feat: support json.RawMessage by @danielgtaylor in #350
- docs: fixed typo in reference docs by @brahmlower in #354
- feat: better support for optional/nullable types by @danielgtaylor in #351
- docs: fix middleware header example by @danielgtaylor in #358
New Contributors
- @brahmlower made their first contribution in #354
Full Changelog: v2.11.0...v2.12.0