This doc details what emitters will generate for versioned specs
Single api Version
If there is just one api version in the spec, we will generate the api surface area for that one version.
import "@typespec/versioning" ;
using TypeSpec . Versioning ;
@versioned ( My . Service . Versions )
v2023_11_01 : "2023-11-01" ,
op stableFunctionality ( @body stableModel : StableModel ) : void ;
from my.service import MyServiceClient, models
client = MyServiceClient( endpoint = ... , credential = ... )
# client's api_version will be "2023-11-01"
stable_model = models.StableModel( stable_feature = "present" )
client.stable_functionality(stable_model) # call goes through
preview_client = MyServiceClient( endpoint = ... , credential = ... , api_version = "2023-11-01" )
# python allows you to override the api version, even if only one version is defined in the spec
public enum ServiceVersion
/// < summary > Service version "2023-11-01". </ summary >
Uri endpoint = new Uri ( "<https://my-service.azure.com>" );
ServiceClient client = new ServiceClient (endpoint);
//client's api_version will be "2023-11-01"
StableModel stableModel = new StableModel ( "<stableFeature>" );
Response response = client. StableFunctionality (stableModel);
Uri endpoint = new Uri ( "<https://my-service.azure.com>" );
ServiceClientOptions options = new ServiceClientOptions (ServiceVersion.V2023_11_01);
ServiceClient client = new ServiceClient (endpoint, options);
//You can specify the service api-version when create client instance. Now client's api_version will be "2023-11-01"
// there's no apiVersion defined in the all the operations, TypeScript emitter will ignore it.
public enum ServiceServiceVersion implements ServiceVersion {
V2023_11_01 ( "2023-11-01" );
public static ServiceServiceVersion getLatest () {} // V2023_11_01
ServiceClientClient serviceClientClient = new ServiceClientClientBuilder ()
// override the api version, even if only one version is defined in the spec
. serviceVersion (ServiceServiceVersion.V2023_11_01)
// client's api-version will be 2023-11-01
StableModel stableModel = new StableModel ( "present" );
serviceClientClient. stableFunctionality (stableModel);
Multiple api versions
The configuration flag api-version
allows you to toggle the behavior that our emitters will generate.
We will get the versioning information from the Versions
enum that you pass to the @versioned
decorator from the @typespec/versioning
library.
NOTE: The ordering of the values in the Versions
enum is very important. We use this information to determine the order of versions. Our default value will be the last entry in the Versions
list
Default
By default our emitters will only generate the surface used by the latest api version if there are multiple defined. This includes generating only the models used in the surface area of the latest api version.
Documentation and enums showing the available api versions will still include all of the known api versions, meaning there will be documentation for both the preview and stable releases.
For the below example, all languages will generate the api surface of default version v2023_11_01
. There will be no generation of the operation previewFunctionality
, and we will also not generate the PreviewModel
because it’s only used in previewFunctionality
, and therefore is not used in the api surface of v2023_11_01
.
import "@typespec/versioning" ;
using TypeSpec . Versioning ;
@versioned ( My . Service . Versions )
v2023_11_01_preview : "2023-11-01-preview" ,
v2023_11_01 : "2023-11-01" ,
@added ( Versions . v2023_11_01_preview )
@removed ( Versions . v2023_11_01 )
op previewFunctionality ( @body previewModel : PreviewModel ) : void ;
op stableFunctionality ( @body stableModel : StableModel ) : void ;
from my.service import MyServiceClient, models
client = MyServiceClient( endpoint = ... , credential = ... )
# client's api_version will be "2023-11-01"
stable_model = models.StableModel( stable_feature = "present" )
client.stable_functionality(stable_model) # call goes through
with pytest.expect( ImportError ):
preview_model = models.PreviewModel( preview_functionality = "not present" )
with pytest.expect( AttributeError ):
client.preview_functionality({ "previewFunctionality" : "not present" })
public enum ServiceVersion
/// < summary > Service version "2023-11-01-preview". </ summary >
/// < summary > Service version "2023-11-01". </ summary >
Uri endpoint = new Uri ( "<https://my-service.azure.com>" );
ServiceClient client = new ServiceClient (endpoint);
//client's api-version will be "2023-11-01"
StableModel stableModel = new StableModel ( "<stableFeature>" );
Response response = client. StableFunctionality (stableModel);
//neither PreviewModel nor PreviewFunctionality will be generated
// there is no apiVersion parameters defined in all operations, TypeScript emitter will ignore it.
public enum ServiceServiceVersion implements ServiceVersion {
V2023_11_01 ( "2023-11-01" );
public static ServiceServiceVersion getLatest () {} // V2023_11_01
ServiceClientClient serviceClientClient = new ServiceClientClientBuilder ()
// client's api-version will be 2023-11-01
StableModel stableModel = new StableModel ( "present" );
serviceClientClient. stableFunctionality (stableModel);
// neither PreviewModel nor previewFunctionality will be generated
Override to a specific version
You can override the signature to return the api surface area for a specific api version.
In this example, you can see how this change is made in tspconfig.yaml, and we are going to override to return the preview api surface area for our spec. The preview api surface area contains all of the functionality.
"@azure-tools/typespec-python" :
api-version : "2023-11-01-preview"
"@azure-tools/typespec-csharp" :
api-version : "2023-11-01-preview"
"@azure-tools/typespec-ts" :
api-version : "2023-11-01-preview"
"@azure-tools/typespec-java" :
api-version : "2023-11-01-preview"
"@azure-tools/typespec-go" :
api-version : "2023-11-01-preview"
import "@typespec/versioning" ;
using TypeSpec . Versioning ;
@versioned ( My . Service . Versions )
v2023_11_01_preview : "2023-11-01-preview" ,
v2023_11_01 : "2023-11-01" ,
@added ( Versions . v2023_11_01_preview )
@removed ( Versions . v2023_11_01 )
op previewFunctionality ( @body previewModel : PreviewModel ) : void ;
op stableFunctionality ( @body stableModel : StableModel ) : void ;
from my.service import MyServiceClient, models
preview_client = MyServiceClient( endpoint = ... , credential = ... )
# client's api_version will be "2023-11-01-preview"
stable_model = models.StableModel( stable_feature = "present" )
preview_client.stable_functionality(stable_model) # call goes through
preview_model = models.PreviewModel( preview_functionality = "present" )
# the model is generated as part of the api surface
preview_client.preview_functionality(preview_model) # call goes through
public enum ServiceVersion
/// < summary > Service version "2023-11-01-preview". </ summary >
Uri endpoint = new Uri ( "<https://my-service.azure.com>" );
ServiceClient client = new ServiceClient (endpoint);
// client's api-version will be "2023-11-01-preview"
//call PreviewFunctionality
PreviewModel previewModel = new PreviewModel ( "<betaFeature>" );
Response response = client. PreviewFunctionality (previewModel);
//call StableFunctionality
StableModel stableModel = new StableModel ( "<stableFeature>" );
Response response = client. StableFunctionality (stableModel);
// there is no apiVersion parameters defined in all operations, TypeScript emitter will ignore it.
public enum ServiceServiceVersion implements ServiceVersion {
V2023_11_01_PREVIEW ( "2023-11-01-preview" );
public static ServiceServiceVersion getLatest () {} // V2023_11_01_PREVIEW
ServiceClientClient serviceClientClient = new ServiceClientClientBuilder ()
// client's api-version will be 2023-11-01-preview
StableModel stableModel = new StableModel ( "present" );
serviceClientClient. stableFunctionality (stableModel);
PreviewModel previewModel = new PreviewModel ( "present" );
serviceClientClient. previewFunctionality (previewModel);
Override to return all
You can also override the signature to return the combined api surface area of all of the separate api versions. Different languages have different support for versioning validation.
In the following examples, you can observe how this change is made in tspconfig.yaml
.
"@azure-tools/typespec-python" :
api-version : "2023-11-01-preview"
"@azure-tools/typespec-csharp" :
api-version : "2023-11-01-preview"
"@azure-tools/typespec-ts" :
api-version : "2023-11-01-preview"
"@azure-tools/typespec-java" :
api-version : "2023-11-01-preview"
"@azure-tools/typespec-go" :
api-version : "2023-11-01-preview"
import "@typespec/versioning" ;
using TypeSpec . Versioning ;
@versioned ( My . Service . Versions )
v2023_11_01_preview : "2023-11-01-preview" ,
v2023_11_01 : "2023-11-01" ,
@added ( Versions . v2023_11_01_preview )
@removed ( Versions . v2023_11_01 )
op previewFunctionality ( @body previewModel : PreviewModel ) : void ;
op stableFunctionality ( @body stableModel : StableModel ) : void ;
from my.service import MyServiceClient, models
client = MyServiceClient( endpoint = ... , credential = ... )
# client's api_version will be "2023-11-01"
stable_model = models.StableModel( stable_feature = "present" )
client.stable_functionality(stable_model) # call goes through
preview_model = models.PreviewModel( preview_functionality = "present" )
# the model is generated as part of the api surface
with pytest.expect( ValueError ) as ex:
client.preview_functionality(preview_model)
assert "preview_functionality is not available in api version 2023-11-01" in str (ex)
preview_client = MyServiceClient( endpoint = ... , credential = ... , api_version = "2023-11-01-preview" )
preview_client.preview_functionality(preview_model) # call goes through
public enum ServiceVersion
/// < summary > Service version "2023-11-01-preview". </ summary >
/// < summary > Service version "2023-11-01". </ summary >
Uri endpoint = new Uri ( "<https://my-service.azure.com>" );
ServiceClient client = new ServiceClient (endpoint);
//client's api_version will be "2023-11-01"
//call PreviewFunctionality
PreviewModel previewModel = new PreviewModel ( "<betaFeature>" );
Response response = client. PreviewFunctionality (previewModel);
//call StableFunctionality
StableModel stableModel = new StableModel ( "<stableFeature>" );
Response response = client. StableFunctionality (stableModel);
Uri endpoint = new Uri ( "<https://my-service.azure.com>" );
ServiceClientOptions options = new ServiceClientOptions (ServiceVersion.V2023_11_01_Preview);
ServiceClient client = new ServiceClient (endpoint, options);
//You can specify the service api-version when create client instance. Now client's api_version will be "2023-11-01-preview"
// there is no apiVersion parameters defined in all operations, TypeScript emitter will ignore it.
Overriding the Client Api Version Parameter
By default, we find api version parameters in specs based off of names. There is special logic we do with api version parameters:
These api version parameters get elevated up to the client level (if the service is versioned)
We auto-add api version information to next links when paging
We set the client default for these parameters to be the default api version for your service.
There are cases where you have an api-versioning parameter without the explicit name api-version
. In these cases, you can use the @isApiVersion
decorator to override and explicitly say whether that parameter is an api version param or not.
import "@typespec/versioning" ;
import "@azure-tools/typespec-client-generator-core" ;
using TypeSpec . Versioning ;
using Azure . ClientGenerator . Core ;
@versioned ( My . Service . Versions )
v2023_11_01 : "2023-11-01" ,
v2024_04_01 : "2024-04-01" ,
from my.service import MyServiceClient
client = MyServiceClient( endpoint = ... , credential = ... )
print (client.version) # == "2024-04-01", since that is the default
client_with_specified_api_version = MyServiceClient( endpoint = ... , credential = ... , version = "2023-11-01" )
print (client.version) # == "2023-11-01", since we specified
retval = client.get() # version is elevated onto the client
public enum ServiceVersion
/// < summary > Service version "2023-11-01". </ summary >
/// < summary > Service version "2024-04-01". </ summary >
Uri endpoint = new Uri ( "<https://my-service.azure.com>" );
ServiceClient client = new ServiceClient (endpoint);
//client's version will be "2024-04-01"
ServiceClientOptions options = new ServiceClientOptions (ServiceVersion.V2023_11_01);
ServiceClient clientWithSpecifiedApiVersion = new ServiceClient (endpoint, options);
//client's version will be "2023-11-01"
Response response = client. get (); // version parameter is elevated onto the client
public enum ServiceServiceVersion implements ServiceVersion {
V2023_11_01 ( "2023-11-01" );
V2024_04_01 ( "2024-04-01" );
public static ServiceServiceVersion getLatest () {} // V2024_04_01
ServiceClientClient client = new ServiceClientClientBuilder ()
// client's version will be 2024-04-01
ServiceClientClient clientWithSpecifiedApiVersion = new ServiceClientClientBuilder ()
// override the api version, even if only one version is defined in the spec
. serviceVersion (ServiceServiceVersion.V2023_11_01)
// client's version will be 2023-11-01
client. get (); // version parameter is elevated onto the client