From 4c51eb24826cd62c8b4ff1dab0ebbbaaf818a648 Mon Sep 17 00:00:00 2001 From: hacker1db Date: Sat, 16 Sep 2023 14:40:46 -0700 Subject: [PATCH] Updated the README.md to support Oauth and the Identity Azure SDK. - Generic Oauth Method to support bringing your own Oauth Token - Support for Azure.Identity SDK I steal this from another PR that has not already been merged and built upon it. see pr#105 --- README.md | 113 ++++++++++++++++++++++++++++++++++- azuredevops/v7/connection.go | 106 ++++++++++++++++++++++++++++++++ azuredevops/v7/go.mod | 6 +- azuredevops/v7/go.sum | 90 +++++++++++++++++++++++++++- 4 files changed, 310 insertions(+), 5 deletions(-) diff --git a/README.md b/README.md index 2d686024..447487ec 100644 --- a/README.md +++ b/README.md @@ -3,9 +3,13 @@ [![Go Report Card](https://goreportcard.com/badge/github.com/microsoft/azure-devops-go-api)](https://goreportcard.com/report/github.com/microsoft/azure-devops-go-api) # Azure DevOps Go API + This repository contains Go APIs for interacting with and managing Azure DevOps. ## Get started + +### Authenticate Using Azure PAT + To use the API, establish a connection using a [personal access token](https://docs.microsoft.com/azure/devops/organizations/accounts/use-personal-access-tokens-to-authenticate?view=azure-devops) and the URL to your Azure DevOps organization. Then get a client using the connection and make API calls. ```go @@ -72,14 +76,119 @@ func main() { } ``` +### Authenticate Using Azure AD Access Tokens (OAuth) + +When Authenticating using Azure AD Access Tokens, you can use either a default token credential or a connection chain token credential. +- [Read more here about using Service Principals and Managed identity to call Azure DevOps](https://learn.microsoft.com/en-us/azure/devops/integrate/get-started/authentication/service-principal-managed-identity?toc=%2Fazure%2Fdevops%2Fmarketplace-extensibility%2Ftoc.json&view=azure-devops) + + +#### Using a default token credential +```go + + connection := azuredevops.NewOAuthConnectionDefault(organizationUrl, azuredevops.TokenOptions{}) + ctx := context.Background() + + // Create a client to interact with the Core area + coreClient, err := core.NewClient(ctx, connection) + if err != nil { + log.Fatal(err) + } + + // The rest of the code below remains the same +``` + + +#### Using a connection chain token credential + +```go +package main + +import ( + "context" + "log" + "strconv" + + "github.com/microsoft/azure-devops-go-api/azuredevops/v7" + "github.com/microsoft/azure-devops-go-api/azuredevops/v7/core" +) + +func main() { + organizationUrl := "https://dev.azure.com/MyOrg" // todo: replace value with your organization url + + connection := azuredevops.NewOAuthConnectionChainToken(organizationUrl, azuredevops.TokenOptions{}) + + + ctx := context.Background() + + // Create a client to interact with the Core area + coreClient, err := core.NewClient(ctx, connection) + if err != nil { + log.Fatal(err) + } + + // Get first page of the list of team projects for your organization + responseValue, err := coreClient.GetProjects(ctx, core.GetProjectsArgs{}) + if err != nil { + log.Fatal(err) + } + + index := 0 + for responseValue != nil { + // Log the page of team project names + for _, teamProjectReference := range (*responseValue).Value { + log.Printf("Name[%v] = %v", index, *teamProjectReference.Name) + index++ + } + + // if continuationToken has a value, then there is at least one more page of projects to get + if responseValue.ContinuationToken != "" { + + continuationToken, err := strconv.Atoi(responseValue.ContinuationToken) + if err != nil { + log.Fatal(err) + } + + // Get next page of team projects + projectArgs := core.GetProjectsArgs{ + ContinuationToken: &continuationToken, + } + responseValue, err = coreClient.GetProjects(ctx, projectArgs) + if err != nil { + log.Fatal(err) + } + } else { + responseValue = nil + } + } +} + +``` + + +### Authenticate using NewOauthGenericConnection + +```go + +// if you want to make the http call to get the token yourself, or you want to use a different library to get the token +connection := azuredevops.NewOAuthGenericConnection(organizationUrl, OAuthToken) +ctx := context.Background() + +// Create a client to interact with the Core area +coreClient, err := core.NewClient(ctx, connection) +if err != nil { +log.Fatal(err) +} +// Rest of the code above remains the same + +``` + ## API documentation This Go library provides a thin wrapper around the Azure DevOps REST APIs. See the [Azure DevOps REST API reference](https://docs.microsoft.com/en-us/rest/api/azure/devops/?view=azure-devops-rest-5.1) for details on calling different APIs. - # Contributing -This project welcomes contributions and suggestions. Most contributions require you to agree to a +This project welcomes contributions and suggestions. Most contributions require you to agree to a Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us the rights to use your contribution. For details, visit https://cla.opensource.microsoft.com. diff --git a/azuredevops/v7/connection.go b/azuredevops/v7/connection.go index eb76f53c..2f1dc733 100644 --- a/azuredevops/v7/connection.go +++ b/azuredevops/v7/connection.go @@ -7,6 +7,10 @@ import ( "context" "crypto/tls" "encoding/base64" + "github.com/Azure/azure-sdk-for-go/sdk/azcore" + "github.com/Azure/azure-sdk-for-go/sdk/azcore/policy" + "github.com/Azure/azure-sdk-for-go/sdk/azidentity" + "log" "strings" "sync" "time" @@ -14,6 +18,8 @@ import ( "github.com/google/uuid" ) +const azureManagementUrl = "https://management.core.windows.net/" + // Creates a new Azure DevOps connection instance using a personal access token. func NewPatConnection(organizationUrl string, personalAccessToken string) *Connection { authorizationString := CreateBasicAuthHeaderValue("", personalAccessToken) @@ -33,6 +39,106 @@ func NewAnonymousConnection(organizationUrl string) *Connection { } } +// NewOAuthGenericConnection Creates a new Azure DevOps connection instance using an OAuth token. +func NewOAuthGenericConnection(organizationUrl string, OAuthToken string) *Connection { + authorizationString := CreateOAuthHeaderValue(OAuthToken) + organizationUrl = normalizeUrl(organizationUrl) + return &Connection{ + AuthorizationString: authorizationString, + BaseUrl: organizationUrl, + SuppressFedAuthRedirect: true, + } +} + +func NewOAuthConnectionDefault(organizationUrl string, defaultOpts TokenOptions) *Connection { + scopes := []string{azureManagementUrl} // This scope is required to get an Azure Ad Access Token, so you can log in to Azure DevOps + defaultCred, err := azidentity.NewDefaultAzureCredential(defaultOpts.defaultOpts) + if err != nil { + log.Fatal(err) + } + var adToken, _ = defaultCred.GetToken(context.Background(), policy.TokenRequestOptions{Scopes: scopes}) + if err != nil { + log.Fatal(err) + } + + authorizationString := CreateOAuthHeaderValue(adToken.Token) + organizationUrl = normalizeUrl(organizationUrl) + return &Connection{ + AuthorizationString: authorizationString, + BaseUrl: organizationUrl, + SuppressFedAuthRedirect: true, + } +} + +func NewOAuthConnectionChainToken(organizationUrl string, tokenOpts TokenOptions) *Connection { + + OAuthToken := requestChainAccessToken(tokenOpts) + authorizationString := CreateOAuthHeaderValue(OAuthToken.Token) + organizationUrl = normalizeUrl(organizationUrl) + return &Connection{ + AuthorizationString: authorizationString, + BaseUrl: organizationUrl, + SuppressFedAuthRedirect: true, + } +} + +func CreateOAuthHeaderValue(token string) string { + return "Bearer " + token +} + +// requestAzureChainCredential Create an Azure AD Access Token using the chained credential azure sdk returns a chained credential +func requestAzureChainCredential(opts TokenOptions) *azidentity.ChainedTokenCredential { + + azCLICred, err := azidentity.NewAzureCLICredential(opts.azCliOpts) + if err != nil { + log.Fatal(err) + } + + mangedCred, err := azidentity.NewManagedIdentityCredential(opts.managedOpts) + if err != nil { + log.Fatal(err) + } + + // you will need the AZURE_TENANT_ID, AZURE_CLIENT_ID, AZURE_CLIENT_SECRET environment variables set for this to work correctly + environmentCred, err := azidentity.NewEnvironmentCredential(opts.envOpts) + + chainCred, err := azidentity.NewChainedTokenCredential([]azcore.TokenCredential{azCLICred, mangedCred, environmentCred}, nil) + if err != nil { + log.Fatal(err) + } + + return chainCred + +} + +// RequestChainAccessToken Creates an Azure AD Access Token using the chained credential azure sdk +func requestChainAccessToken(tokenOptions TokenOptions) azcore.AccessToken { + + chainCred := requestAzureChainCredential(tokenOptions) + + // Scopes can only have one element in the array, so we are hard coding the scope + scopes := []string{azureManagementUrl} // This scope is required to get an Azure Ad Access Token, so you can log in to Azure DevOps + + //var adToken, err = chainCred.GetToken(context.Background(), policy.TokenRequestOptions{Scopes: scopes}) + var adToken, err = chainCred.GetToken(context.Background(), policy.TokenRequestOptions{Scopes: scopes}) + if err != nil { + log.Fatal(err) + } + + return adToken +} + +type TokenOptions struct { + azCliOpts *azidentity.AzureCLICredentialOptions + managedOpts *azidentity.ManagedIdentityCredentialOptions + envOpts *azidentity.EnvironmentCredentialOptions + defaultOpts *azidentity.DefaultAzureCredentialOptions +} + +func NewTokenOptions() *TokenOptions { + return &TokenOptions{envOpts: nil, managedOpts: nil, azCliOpts: nil} +} + type Connection struct { AuthorizationString string BaseUrl string diff --git a/azuredevops/v7/go.mod b/azuredevops/v7/go.mod index 0c3ebc6c..87e21f08 100644 --- a/azuredevops/v7/go.mod +++ b/azuredevops/v7/go.mod @@ -2,4 +2,8 @@ module github.com/microsoft/azure-devops-go-api/azuredevops/v7 go 1.12 -require github.com/google/uuid v1.1.1 +require ( + github.com/Azure/azure-sdk-for-go/sdk/azcore v1.7.2 + github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.3.1 + github.com/google/uuid v1.3.0 +) diff --git a/azuredevops/v7/go.sum b/azuredevops/v7/go.sum index b864886e..ca486c82 100644 --- a/azuredevops/v7/go.sum +++ b/azuredevops/v7/go.sum @@ -1,2 +1,88 @@ -github.com/google/uuid v1.1.1 h1:Gkbcsh/GbpXz7lPftLA3P6TYMwjCLYm83jiFQZF/3gY= -github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/Azure/azure-sdk-for-go/sdk/azcore v1.7.1/go.mod h1:bjGvMhVMb+EEm3VRNQawDMUyMMjo+S5ewNjflkep/0Q= +github.com/Azure/azure-sdk-for-go/sdk/azcore v1.7.2 h1:t5+QXLCK9SVi0PPdaY0PrFvYUo24KwA0QwxnaHRSVd4= +github.com/Azure/azure-sdk-for-go/sdk/azcore v1.7.2/go.mod h1:bjGvMhVMb+EEm3VRNQawDMUyMMjo+S5ewNjflkep/0Q= +github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.3.1 h1:LNHhpdK7hzUcx/k1LIcuh5k7k1LGIWLQfCjaneSj7Fc= +github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.3.1/go.mod h1:uE9zaUfEQT/nbQjVi2IblCG9iaLtZsuYZ8ne+PuQ02M= +github.com/Azure/azure-sdk-for-go/sdk/internal v1.3.0 h1:sXr+ck84g/ZlZUOZiNELInmMgOsuGwdjjVkEIde0OtY= +github.com/Azure/azure-sdk-for-go/sdk/internal v1.3.0/go.mod h1:okt5dMMTOFjX/aovMlrjvvXoPMBVSPzk9185BT0+eZM= +github.com/AzureAD/microsoft-authentication-library-for-go v1.1.1 h1:WpB/QDNLpMw72xHJc34BNNykqSOeEJDAWkhf0u12/Jk= +github.com/AzureAD/microsoft-authentication-library-for-go v1.1.1/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/dnaeon/go-vcr v1.1.0/go.mod h1:M7tiix8f0r6mKKJ3Yq/kqU1OYf3MnfmBWVbPx/yU9ko= +github.com/dnaeon/go-vcr v1.2.0 h1:zHCHvJYTMh1N7xnV7zf1m1GPBF9Ad0Jk/whtQ1663qI= +github.com/dnaeon/go-vcr v1.2.0/go.mod h1:R4UdLID7HZT3taECzJs4YgbbH6PIGXB6W/sc5OLb6RQ= +github.com/golang-jwt/jwt/v5 v5.0.0 h1:1n1XNM9hk7O9mnQoNBGolZvzebBQ7p93ULHRc28XJUE= +github.com/golang-jwt/jwt/v5 v5.0.0/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk= +github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= +github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= +github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= +github.com/modocache/gover v0.0.0-20171022184752-b58185e213c5/go.mod h1:caMODM3PzxT8aQXRPkAt8xlV/e7d7w8GM5g0fa5F0D8= +github.com/montanaflynn/stats v0.7.0/go.mod h1:etXPPgVO6n31NxCd9KQUMvCM+ve0ruNzt6R8Bnaayow= +github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8 h1:KoWmjvw+nsYOo29YJK9vDA65RGE3NrOnUtO7a+RF9HU= +github.com/pkg/browser v0.0.0-20210911075715-681adbf594b8/go.mod h1:HKlIX3XHQyzLZPlr7++PzdhaXEj94dEiJgZDTsxEqUI= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= +golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.12.0 h1:tFM/ta59kqch6LlvYnPa0yx5a83cL2nHflFhYKvv9Yk= +golang.org/x/crypto v0.12.0/go.mod h1:NF0Gs7EO5K4qLn+Ylc+fih8BSTeIjAP05siRnAh98yw= +golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= +golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= +golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= +golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= +golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= +golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc= +golang.org/x/net v0.10.0/go.mod h1:0qNGK6F8kojg2nk9dLZ2mShWaEBan6FAoqfSigmmuDg= +golang.org/x/net v0.14.0 h1:BONx9s002vGdD9umnlX1Po8vOZmrgH34qlHcD1MfK14= +golang.org/x/net v0.14.0/go.mod h1:PpSgVXXLK0OxS0F31C1/tv6XNguvCrnXIDrFMspZIUI= +golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210616045830-e2b7044e8c71/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.11.0 h1:eG7RXZHdqOJ1i+0lgLgCpSXAp6M3LYlAo6osgSi0xOM= +golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= +golang.org/x/term v0.6.0/go.mod h1:m6U89DPEgQRMq3DNkDClhWw02AUbt2daBVO4cn4Hv9U= +golang.org/x/term v0.8.0/go.mod h1:xPskH00ivmX89bAKVGSKKtLOWNx2+17Eiy94tnKShWo= +golang.org/x/term v0.11.0/go.mod h1:zC9APTIj3jG3FdV/Ons+XE1riIZXG4aZ4GTHiPZJPIU= +golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= +golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= +golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/text v0.9.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/text v0.12.0 h1:k+n5B8goJNdU7hSvEtMUz3d1Q6D/XW4COJSJR6fN0mc= +golang.org/x/text v0.12.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= +golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU= +golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= +gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=