All notable changes to this project will be documented in this file. Please follow the Keep a Changelog standard.
- Fix invalid migration of pydantic v1 style root validators when pydantic erases information about their "skip_on_failure" attribute
- Type hints for newest pydantic versions
- Bumped package versions
- Non-function (object instances) dependencies not supporting dependency overrides in testing
- Python 3.13 to CI
- Support for webhooks in swagger
- Automatic generation of versioned routes and webhooks upon the first request to Cadwyn. Notice that if you were using some of cadwyn's internal interfaces, this might break your code. If it did, make an issue and let's discuss your use case
- Removed typer from main dependencies
- Migrated from poetry to uv (Contributed by @bastienpo)
- Removed CLI and Uvicorn dependencies from Cadwyn installations by default. Added a
standard
extras group to mirror FastAPI (Contributed by @bastienpo)
- Background tasks not functioning in versioned endpoints
- Added support for
embed_body_fields
insolve_dependencies
andcreate_model_field
in FastAPI. FastAPI has made a breaking change for these interfaces which is why we had to fix it - Fixed invalid imports in quickstart docs
- Fixed default dependencies not including the CLI for FastAPI, thus causing the quickstart docs to be invalid
- FastAPI Servers list did not include root path when mounted as a sub-app unless specified directly within the user defined server list (Contributed by @OD-tpeko)
- OpenAPI spec did not include the
summary
field (Contributed by @OD-tpeko)
- Previously the docs were showing wrong versioned doc paths when cadwyn was mounted as a sub-app (Contributed by @OD-tpeko)
- Automatic changelog generation from version changes using
Cadwyn.generate_changelog
method andGET /changelog
endpoint. - Automatic creation of versioned_routers based on the
VersionBundle
passed toCadwyn
which means that all versions mentioned in the version bundle will already be available for routing even without the use ofgenerate_and_include_versioned_routers
- Renamed
Version.version_changes
toVersion.changes
regex
,include
,min_items
,max_items
, andunique_items
arguments were removed fromschema(...).field(...).had
. Notice that it's not a breaking change for most cases because passing these arguments caused exceptions
- Exposed
cadwyn.generate_versioned_models
that allow you to generate pydantic models and enums even without a FastAPI app
Versions 3.x.x are still supported in terms of bug and security fixes but all the new features will go into versions 4.x.x.
- Runtime schema/enum generation
- Support for versions as ISO date strings instead of dates in
cadwyn.migrate_response_body
andcadwyn.Version
- Pydantic 1 support
- Code generation from everywhere. It is now completely replaced by runtime generation (so schemas/enums are generated in the same manner as endpoints). This allows Cadwyn to version things outside of your project and allows you to pick any project structure unlike codegen that required a single "head" directory with all the versioned modules.
- CLI commands for codegen
cadwyn.main
because it is replaced bycadwyn.__init__
cadwyn.structure.module
as it was only necessary in codegencadwyn.VersionBundle.latest_schemas_package
andcadwyn.VersionBundle.head_schemas_package
,cadwyn.VersionBundle.versioned_modules
,cadwyn.VersionBundle.versioned_directories_with_head
,cadwyn.VersionBundle.versioned_directories_without_head
, because they were only necessary in code generationcadwyn.Cadwyn.add_unversioned_routers
as you can now simply use FastAPI'sinclude_router
cadwyn.Cadwyn.add_unversioned_routes
as you can now simply use any of FastAPI's methods for adding routes directly to the appcadwyn.Cadwyn.enrich_swagger
as its functionality has been automatedcadwyn.InternalRepresentationOf
as it was deprecated previously and is now replaced with HeadVersion migrations
VersionBundle.migrate_response_body
is no longer a method ofVersionBundle
and is now importable directly fromcadwyn
as a functioncadwyn.structure
is no longer recommended to be used directly because everything from it is now available incadwyn
directly
- Invalid unparseable JSON response without quotes when the response was a raw string JSONResponse
- An exception raised during codegen if a pydantic model or its parent were created within some indent such as classes defined under if statements
- Wrong globals being used for wrapped endpoints in older versions, sometimes causing FastAPI to fail to resolve forward references on endpoint generation (see #192 for more details)
- dependency_overrides not working for old versions of the endpoints because they were wrapped and wraps did not have the same
__hash__
and__eq__
as the original dependencies
HeadVersion
andVersion
intocadwyn.__init__
so now they are directly importable fromcadwyn
- Fix dependencies from other libraries not resolving if they use fastapi.Request or fastapi.Response (added svcs-specific test)
- Now a proper exception is used when no dated version was passed into
VersionBundle
- Fix dependency overrides not working for versioned routes
Cadwyn.enrich_swagger
is now completely unnecessary: openapi is now generated at runtime. It also now does not do anything, is deprecated, and will be removed in a future version
- fastapi-pagination now does not require an explicit call to
Cadwyn.enrich_swagger
Cadwyn.router.routes
now includes all existing routers within the application instead of just unversioned routes
- Compatibility with fastapi-pagination
- High cardinality of metrics for routers with path variables in starlette-exporter
- Openapi not being generated when lifespan was used
- Badge links in the readme
- Optimized the calls to enrich_swagger which now happen on the startup event, once for the whole application
- Oauth2 authentication parameters did not get passed to swagger
- Current API version to per-version openapi.json
- Validation for path converters to make sure that impossible HTTP methods cannot be used
- Validation for both path and schema converters to make sure that they are used at some point. Otherwise, router generation will raise an error
fastapi.Response
subclasses with non-null bodies and 500 response causing the response to not get returnedfastapi.Response
subclasses had invalid content length if migration affected it
- Rewritten header routing logic and structure to support the full feature set of FastAPI
- Modules and enums from head versions not being detected and thus causing errors
- Header router is no longer reliant on the API version header -- now it simply takes the API version from the
VersionBundle.api_version_var
, thus making it easy for someone to extend header routing and set their own rules for how the default version is chosen
- Previous version introduced a minor breaking change: if any old users depended on the pure
generate_versioned_routers
interface, their work would receive a minor yet simple breaking change.
Yanked due to a minor breaking change that we fixed in 3.10.1.
- The new approach to internal schemas: instead of having them duplicate certain fields from
latest
, we introduced a newHEAD
version -- the only one the user maintains by hand. All requests get migrated toHEAD
and latest schemas are generated fromHEAD
.cadwyn.structure.HeadVersion
was added to give us the ability to have migrations betweenHEAD
and latest, thus eliminating the need forInternalRepresentationOf
because all the used schemas are now the internal representations
latest
is now namedhead
because it no longer represents the newest version. Instead, it is the theinternally used
version and the version that is used for generating all other versions.- the newest version is not aliased from
latest
anymore. Instead, it is generated like all the rest - deprecated
InternalRepresentationOf
and the concept ofinternal schemas
in favor ofHeadVersion
migrations
- A broken link to docs in README.md
- Support for getting openapi.json routes using API version headers instead of path query params
- Discord status badge in README
- Logos to existing status badges in README
- An ability to specify multiple schemas when using
convert_request_to_next_version_for
andconvert_response_to_next_version_for
to be able to migrate multiple types of schemas using the same converter - Redoc support
- Dependency from verselect. Now it is included as a part of Cadwyn
h11._util.LocalProtocolError
when raisingHTTPException(status_code=500)
- Error message for changing path params of an endpoint in an incompatible manner which listed methods instead of path params
- Deprecated
cadwyn generate-code-for-versioned-packages
and addedcadwyn codegen
instead. It doesn't requiretemplate_package
argument anymore and does not have theignore_coverage_for_latest_aliases
argument as we plan to remove this feature in the future. So it only requiresversion_bundle
.
- Deprecated
cadwyn generate-code-for-versioned-packages
and addedcadwyn codegen
instead. It doesn't requiretemplate_package
argument anymore and does not have theignore_coverage_for_latest_aliases
argument as we plan to remove this feature in the future. So it only requiresversion_bundle
.
- When a class-based dependency from fastapi was used (anything security related), FastAPI had hardcoded
isinstance
checks for it which it used to enrich swagger with functionality. But when the dependencies were wrapped into our function wrappers, these checks stopped passing, thus breaking this functionality in swagger. Now we ignore all dependencies that FastAPI creates. This also introduces a hard-to-solve bug: if fastapi's class-based security dependency was subclassed and then__call__
was overridden with new dependencies that are versioned -- we will not migrate them from version to version. I hope this is an extremely rare use case though. In fact, such use case breaks Liskov Substitution Principle and doesn't make much sense because security classes already includerequest
parameter which means that no extra dependencies or parameters are necessary.
- When a class-based dependency was used, its dependant was incorrectly generated, causing all affected endpoints to completely stop functioning
- A rare pydantic 2 bug that caused
BaseModel
annotations to be corrupted when new fields were added to the schema
- Removed exception when creating
cadwyn.Cadwyn
withoutlatest_schemas_package
as it was a minor breaking change
- Add
cadwyn.VersionBundle.migrate_response_body
that allows us to migrate response bodies outside of routing and FastAPI latest_schemas_package
argument tocadwyn.VersionBundle
to support the migration above
- We now raise a 5xx error (
cadwyn.exceptions.CadwynLatestRequestValidationError
) whenever a request migration caused our payload to be incompatible with latest request schemas - Deprecated
cadwyn.main
and usecadwyn.applications
instead - Deprecated
latest_schemas_package
argument incadwyn.Cadwyn
- Previously, Cadwyn did not set the default status code for ResponseInfo
- HTTP status error handling in response converters using
convert_response_to_previous_version_for(..., migrate_http_errors=True)
- Request and response converters were not applied when path params were present
RouterPathParamsModifiedError
is now raised ifendpoint(...).had(path=...)
has different path params than the original route
- Fix import aliases in nested
__init__.py
files generating incorrectly for latest version
- If the endpoint specified a single non-pydantic (list/dict) body parameter, Cadwyn failed to serialize the body
schema(...).validator(...).existed
andschema(...).validator(...).didnt_exist
instructions for simplistic manipulation of validators- Automatic deletion of validators when the fields they validate get deleted
schema(...).field(...).didnt_have
for unsetting field attributes- Improved support for
typing.Annotated
in schemas - Full preservation of original abstract syntax trees for all field values and annotations
- If the user wrote a wrong signature in a transformer decorated by
convert_request_to_next_version_for
orconvert_response_to_previous_version_for
, the text of the error suggested the wrong argument count and names
- Added backwards compatibility for FastAPI < 0.106.0
- Guaranteed that it is impossible to release cadwyn with the wrong pydantic dependency
- Downgrade required version of verselect for backwards compatibility
- Removed lazy migrations as they were producing incorrect results when there were no migrations but when there were schema changes
- Added compatibility with fastapi>=0.109.0
- If a user used a FastAPI/Starlette
StreamingResponse
orFileResponse
, we still tried to access itsbody
attribute which caused anAttributeError
- Sponsors section to README and docs, along with Monite as our main and only current sponsor ✨
- Switched to
better-ast-comments
becauseast-comments
had no license listed on pypi (even though its actual license was MIT) which caused some dependency checking tools to report it as unlicensed
- Migrate from black to ruff-format
- A rare Pydantic 2 bug in internal body schema handling when it was applied too early, causing partially incomplete data to arrive to the handler
- Previously we did not pass
dependency_overrides_provider
,response_model_exclude_unset
response_model_exclude_defaults
, andresponse_model_exclude_none
tofastapi
which could cause erroneous behaviour during serialization in rare cases.
module(...).had(import_=...)
construct for adding imports in older versions- Codegen plugin system that allows easily customizing code generation for any purpose. It also significantly simplifies the core code of code generation
- If a user returned a FastAPI/Starlette
Response
with an empty body, we still tried to serialize it which caused an invalid response body
- Pydantic 2 support
- Expanded reference section to docs
- Contributor docs
- Expanded makefile commands
- internal request representation is now done using an annotation
latest_schemas_module
was renamed tolatest_schemas_package
everywhereapi_version_var
inVersionBundle
is now an optional argument instead of a required one
cadwyn.internal_body_representation_of
because it is now done using an annotation
schema(...).field(...).had(ge=...)
for union fields previously raised anAttributeError
on code generation
- Field ASTs not preserving the original structure when constrained fields were changed
- Support for synchronous endpoints
- The bug where fields from parent schemas also appeared in child schemas
- Migrate from external verselect to stable verselect
- Previously whenever we generated routes, we wrapped all endpoints and callbacks in decorators for every version which caused stacktraces to be unnecessarily huge. Now there is only one wrapper for all versions
cadwyn.Cadwyn
class similar tofastapi.FastAPI
that provides header routing and encapsulates all information about versioned routes- Migrated from
fastapi-header-routing
toverselect
cadwyn.routing.VERSION_HEADER_FORMAT
fromverselect.routing
*versions
argument incadwyn.VersionBundle
is now split into a required positional-onlylatest_version
and*other_versions
to make it possible to see an invalid versionless definition statically. Note that it is not a breaking change because the presence of at least one version was also implicitly required before and would produce a failure at runtime
cadwyn.get_cadwyn_dependency
andcadwyn.header
because it is fully replaced withverselect
- Validation for the spelling of HTTP methods in
cadwyn.structure.endpoint
.
- A prototype of AST-based code generation where we try to keep as much of the old field/annotation structure as possible
UploadFile
and forms have previously caused exceptions on request migration
ContextVar[datetime.date]
being incompatible withVersionBundle
- A note into reference docs about the paths specification in CLI
- Custom body fields created by fastapi caused an exception. Now they are ignored
- A link to openapi discussion on enum expansion into docs/recipes
- A link to intercom's API versioning article into docs/theory
generate_versioned_routers
did not alterAPIRoute.dependant.call
,APIRoute.response_field
, andAPIRoute.secure_cloned_response_field
before which caused these fields to represent latest version on all generated versions. However, this was only a bug if the routes were later added into the app/router by hand instead of usinginherit_routes
oradd_api_route
.
generate_versioned_routers
now accepts only one router instead of a sequence of routers to give us the ability to guarantee that the type of generated routers is the same as the type of the passed router.
- Theory section to docs
- Recipes documentation section
schema(...).field(...).had(name=...)
functionality to rename fields
- Tutorial example structure in tests
cadwyn.main._Cadwyn
experimental private class for automatically adding the header dependency and managing all objects related to versioning
cadwyn.header_routing
which only had experimental private functions
- Route callbacks didn't get migrated to versions with their parent routes
ignore_coverage_for_latest_aliases
argument togenerate_code_for_versioned_packages
which controls whether we add "a pragma: no cover" comment to the star imports in the generated version of the latest module. It is True by default.
- Add back the approach where the first version being an alias to latest in codegen
- Add current working dir to
sys.path
when running code generation through CLI - Use
exclude_unset
when migrating the body of a request to make sure that users'exclude_unset
logic gets preserved
- Pass first argument in
typer.Argument
to prevent errors on older typer versions
- Command-line interface capable of running code-generation and outputting version info
- Internal request schema which gives us all the functionality we could ever need to migrate request bodies forward without any complexity of the prior solution
_get_versioned_router
and experimental header routing with it (by @tikon93). Note that the interface for this feature will change in the future
- Renamed
cadwyn.regenerate_dir_to_all_versions
tocadwyn.generate_code_for_versioned_packages
- Renamed
cadwyn.generate_all_router_versions
tocadwyn.generate_versioned_routers
unions
directory and all logic around it (replaced by internal request schema)FillablePrivateAttr
and all logic around it (replaced by internal request schema)schema(...).property
constructor and all logic around it (replaced by internal request schema)- Special-casing for code generation of package with latest version using star imports from latest (replaced by internal request schema)