I received an application startup exception, related to the OData API versioning. The exception message:
The entity model (EDM) does not have the required ApiVersionAnnotation annotation
The snippet of mapping versioned odata route
public static IEndpointRouteBuilder MapPaymentsODataRoute(this IEndpointRouteBuilder builder)
{
builder.MapVersionedODataRoute(
"payments-odata",
"api/v{apiVersion:apiVersion}/payment-service/odata",
new List<IEdmModel> { BuildEdmModel() });
return builder;
}
All needed API versioning services registered in DI container.
See endpoint configuration: endpoint configuration
See controller example: controller example
Any thoughts? What's the problem?
It's not entirely clear what
BuildEdmModel()
does, but it would appear that you are building it manually instead of via the VersionedODataModelBuilder. This is allowed, but know that you need to set the ApiVersionAnnotation. API Versioning for OData uses an EDM per API Version approach, which keeps everything clearly segregated. API Versioning determines how to match up the EDM by comparing the applied ApiVersionAnnotation with the resolved, incoming API Version. "There can be only one."If you're building the EDMs yourself, you need to add this required annotation to the model itself:
If you only have a single version, this is pretty easy to build. If you have several versions, it gets more complex. The VersionedODataModelBuilder achieves this by creating a unioned set of all API versions, from all OData APIs, discovered through IActionDescriptorCollectionProvider. For every discovered OData API version, a set of IModelConfiguration instances are used to apply the configuration for that particular EDM in a particular API Version, including nothing at all (e.g. excluded). This is determined by the implementer (e.g. developer). Depending on the complexity of your application, you might use a single IModelConfiguration, one per entity, or none at all and just use the
VersionedODataModelBuilder.DefaultModelConfiguration
callback.Regardless of the approach you use, if the ApiVersionAnnotation is set, you'll be back in business.
UPDATE: In response to the added images.
There are several additional points:
MapVersionedODataRoute
as a parameter, it will do the right thing. Due to OData's prefix concept, the IModelConfiguration can vary by API version and path. Alternatively, you can callVersionedODataModelBuilder.GetEdmModels('path')
. If there are no differences between the prefixes, then you're fine.controller
route parameter, then it should work. If not, then your controller name needs to match or you can use the provided[ControllerName("payments")]
to satisfy the requirement. I assume the name of the corresponding entity set isPayments
.[ApiExplorerSettings(IgnoreApi = false)]
will likely cause errors at runtime. The API Versioning extensions are aware that ODataController explicitly opts out of the API Explorer. By default, it will completely ignore theIgnoreApi
setting so that you don't have to reconfigure and opt in every single ODataController. If you really want to use ApiExplorerSettingsAttribute, then you can configure the API Explorer extensions to honor it withODataApiExplorerOptions.UseApiExplorerSettings
, which defaults tofalse
. If you don't change it totrue
, the setting is completely ignored which may include controllers that you intentionally marked asIgnoreApi = true
.