This document is intended for users that are familiar with the older version of Azure SDKs of management libraries for Go language (package name starts with services/**/mgmt/**
, also known as track1) and pursue to migrate their application code on to the latest version of Azure SDKs for Go that conform to our current Azure SDK guidelines (SDK package name starts with sdk/resourcemanager/**/arm**
, also known as track2). The older Golang libraries that do not conform to Azure SDK guidelines are deprecated by 30 September 2023. Although the older libraries can still be used beyond deprecation, they will no longer receive official support and updates from Microsoft after that date.
For users new to the Azure SDK For Go for resource management modules, please see the README for 'sdk/azcore` and the README for every individual package.
- Go 1.18
- Latest version of resource management modules
The latest Azure SDK For Go for management modules is using the Go Modules to manage the dependencies. We ship every RP as an individual module to create a more flexible user experience. Unlike previous version, the latest SDK only provides one API version in one module for each RP. The SDK stable version will always use the service's stable API version. You can search in pkg.go.dev to find the preview version and specify version num when using go get
to retrive it (e.g., go get github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/keyvault/[email protected]
).
In the previous version (services/**/mgmt/**
), autorest.Authorizer
is used in authentication process.
In the latest version (sdk/resourcemanager/**/arm**
), in order to provide a unified authentication based on Azure Identity for all Azure Go SDKs, the authentication mechanism has been re-designed and improved to offer a simpler interface.
To the show the code snippets for the change:
Previous version (services/**/mgmt/**
)
import "github.com/Azure/go-autorest/autorest/adal"
import "github.com/Azure/azure-sdk-for-go/services/resources/mgmt/2020-10-01/resources"
authorizer, err := adal.NewServicePrincipalToken(oAuthToken, "<ClientId>", "<ClientSecret>", endpoint)
client := resources.NewGroupsClient("<SubscriptionId>")
client.Authorizer = authorizer
Latest version (sdk/resourcemanager/**/arm**
)
import "github.com/Azure/azure-sdk-for-go/sdk/azidentity"
import "github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resources/armresources"
credential, err := azidentity.NewClientSecretCredential("<TenantId>", "<ClientId>", "<ClientSecret>", nil)
clientFactory, err := armresources.NewClientFactory(<subscription ID>, credential, &options)
if err != nil {
log.Fatal(err)
}
client := clientFactory.NewResourceGroupsClient()
For detailed information on the benefits of using the new authentication types, please refer to this page
In the previous version (services/**/mgmt/**
), there are two functions to initialize the clients.
In the latest version (sdk/resourcemanager/**/arm**
), one initialization function with custom endpoint has removed. If you want to using a custom endpoint. You need to set endpoint with *arm.ClientOptions
param.
Previous version (services/**/mgmt/**
)
import "github.com/Azure/azure-sdk-for-go/services/compute/mgmt/2019-07-01/compute"
client := compute.NewVirtualMachinesClient("<SubscriptionId>")
Latest version (sdk/resourcemanager/**/arm**
)
import "github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/compute/armcompute"
client, err := armcompute.NewVirtualMachinesClient("<SubscriptionId>", credential, nil)
Previous version (services/**/mgmt/**
)
import "github.com/Azure/azure-sdk-for-go/services/compute/mgmt/2019-07-01/compute"
client := compute.NewVirtualMachinesClientWithBaseURI("<Endpoint>", "<SubscriptionId>")
Latest version (sdk/resourcemanager/**/arm**
)
import "github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/compute/armcompute"
options := arm.ClientOptions {
ClientOptions: azcore.ClientOptions {
Cloud: cloud.AzureChina,
},
}
clientFactory, err := armcompute.NewClientFactory(<subscription ID>, credential, &options)
if err != nil {
log.Fatal(err)
}
client := clientFactory.NewVirtualMachinesClient()
For detailed information on the cloud configuration, please refer to this page
There are some minor changes in the error handling.
- When there is an error in the SDK request, in the previous version (
services/**/mgmt/**
), the return value will all be non-nil, and you can get the raw HTTP response from the response value. In the latest version (sdk/resourcemanager/**/arm**
), the first return value will be empty and you need to convert the error to theazcore.ResponseError
interface to get the raw HTTP response. - In the latest version (
sdk/resourcemanager/**/arm**
), you can always get the raw HTTP response from request context regardless of request result.
Previous version (services/**/mgmt/**
)
resp, err := resourceGroupsClient.CreateOrUpdate(context.TODO(), resourceGroupName, resourceGroupParameters)
if err != nil {
log.Fatalf("Status code: %d", resp.Response().StatusCode)
}
Latest version (sdk/resourcemanager/**/arm**
)
import "github.com/Azure/azure-sdk-for-go/sdk/azcore"
import "github.com/Azure/azure-sdk-for-go/sdk/azcore/runtime"
var rawResponse *http.Response
ctx := context.TODO() // your context
ctxWithResp := runtime.WithCaptureResponse(ctx, &rawResponse)
resp, err := resourceGroupsClient.CreateOrUpdate(ctxWithResp, resourceGroupName, resourceGroupParameters, nil)
if err != nil {
// with error, you can get RawResponse from context
log.Printf("Status code: %d", rawResponse.StatusCode)
var respErr *azcore.ResponseError
if errors.As(err, &respErr) {
// with error, you can also get RawResponse from error
log.Fatalf("Status code: %d", respErr.RawResponse.StatusCode)
} else {
log.Fatalf("Other error: %+v", err)
}
}
// without error, you can get RawResponse from context
log.Printf("Status code: %d", rawResponse.StatusCode)
In the previous version, if a request is a long-running operation, a struct **Future
will be returned, which is an extension of the interface azure.FutureAPI
. You need to invoke the future.WaitForCompletionRef
to wait until it finishes.
In the latest version, if a request is a long-running operation, the function name will start with Begin
to indicate this function will return a poller type which contains the polling methods.
Previous version (services/**/mgmt/**
)
ctx := context.TODO() // your context
future, err := virtualMachinesClient.CreateOrUpdate(ctx, "<resource group name>", "<virtual machine name>", param)
if err != nil {
log.Fatal(err)
}
if err := future.WaitForCompletionRef(ctx, virtualMachinesClient.Client); err != nil {
log.Fatal(err)
}
vm, err := future.Result(virtualMachinesClient)
if err != nil {
log.Fatal(err)
}
log.Printf("virtual machine ID: %v", *vm.ID)
Latest version (sdk/resourcemanager/**/arm**
)
poller, err := client.BeginCreateOrUpdate(context.TODO(), "<resource group name>", "<virtual machine name>", param, nil)
if err != nil {
log.Fatal(err)
}
resp, err := poller.PollUntilDone(context.TODO(), nil)
if err != nil {
log.Fatal(err)
}
log.Printf("virtual machine ID: %v", *resp.VirtualMachine.ID)
In the previous version, if a request is a paginated operation, a struct **ResultPage
will be returned, which is a struct with some paging methods but no interfaces are defined regarding that.
In the latest version, if a request is a paginated operation, a struct **Pager
will be returned that contains the paging methods.
Previous version (services/**/mgmt/**
)
pager, err := resourceGroupsClient.List(context.TODO(), "", nil)
if err != nil {
log.Fatal(err)
}
for p.NotDone() {
for _, v := range pager.Values() {
log.Printf("resource group ID: %s\n", *rg.ID)
}
if err := pager.NextWithContext(context.TODO()); err != nil {
log.Fatal(err)
}
}
Latest version (sdk/resourcemanager/**/arm**
)
ctx := context.TODO() // your context
pager := resourceGroupsClient.NewListPager(nil)
for pager.More() {
nextResult, err := pager.NextPage(ctx)
if err != nil {
log.Fatalf("failed to advance page: %v", err)
}
for _, rg := range nextResult.Value {
log.Printf("resource group ID: %s\n", *rg.ID)
}
}
Because of adopting Azure Core which is a shared library across all Azure SDKs, there is also a minor change regarding how customized policy in configured.
In the previous version (services/**/mgmt/**
), we use the (autorest.Client).Sender
, (autorest.Client).RequestInspector
and (autorest.Client).ResponseInspector
properties in github.com/Azure/go-autorest/autorest
module to provide customized interceptor for the HTTP traffic.
In latest version (sdk/resourcemanager/**/arm**
), we use arm.ClientOptions.PerCallPolicies
and arm.ClientOptions.PerRetryPolicies
in github.com/Azure/azure-sdk-for-go/sdk/azcore/arm
package instead to inject customized policy to the pipeline.
Similar to the customized policy, there are changes regarding how the custom HTTP client is configured as well. You can now use the arm.ClientOptions.Transport
option in github.com/Azure/azure-sdk-for-go/sdk/azcore/arm
package to use your own implementation of HTTP client and plug in what they need into the configuration. The HTTP client must implement the policy.Transporter
interface.
Previous version (services/**/mgmt/**
)
import "github.com/Azure/azure-sdk-for-go/services/resources/mgmt/2020-10-01/resources"
httpClient := NewYourOwnHTTPClient{}
client := resources.NewGroupsClient("<SubscriptionId>")
client.Sender = &httpClient
Latest version (sdk/resourcemanager/**/arm**
)
import "github.com/Azure/azure-sdk-for-go/sdk/azcore/arm"
import "github.com/Azure/azure-sdk-for-go/sdk/azcore/policy"
import "github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/resources/armresources"
httpClient := NewYourOwnHTTPClient{}
options := &arm.ClientOptions{
ClientOptions: policy.ClientOptions{
Transport: &httpClient,
},
}
clientFactory, err := armresources.NewClientFactory(<subscription ID>, credential, &options)
if err != nil {
log.Fatal(err)
}
client := clientFactory.NewResourceGroupsClient()
If you have encountered an issue during migration, please file an issue via Github Issues.