mirror of https://github.com/portainer/portainer
				
				
				
			
		
			
				
	
	
		
			292 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			292 lines
		
	
	
		
			7.6 KiB
		
	
	
	
		
			Go
		
	
	
| package git
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	gittypes "github.com/portainer/portainer/api/git/types"
 | |
| 
 | |
| 	_ "github.com/joho/godotenv/autoload"
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| const privateAzureRepoURL = "https://portainer.visualstudio.com/gitops-test/_git/gitops-test"
 | |
| 
 | |
| func TestService_ClonePublicRepository_Azure(t *testing.T) {
 | |
| 	ensureIntegrationTest(t)
 | |
| 
 | |
| 	pat := getRequiredValue(t, "AZURE_DEVOPS_PAT")
 | |
| 	service := NewService(context.TODO())
 | |
| 
 | |
| 	type args struct {
 | |
| 		repositoryURLFormat string
 | |
| 		referenceName       string
 | |
| 		username            string
 | |
| 		password            string
 | |
| 	}
 | |
| 	tests := []struct {
 | |
| 		name    string
 | |
| 		args    args
 | |
| 		wantErr bool
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "Clone Azure DevOps repo branch",
 | |
| 			args: args{
 | |
| 				repositoryURLFormat: "https://:%s@portainer.visualstudio.com/gitops-test/_git/gitops-test",
 | |
| 				referenceName:       "refs/heads/main",
 | |
| 				username:            "",
 | |
| 				password:            pat,
 | |
| 			},
 | |
| 			wantErr: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "Clone Azure DevOps repo tag",
 | |
| 			args: args{
 | |
| 				repositoryURLFormat: "https://:%s@portainer.visualstudio.com/gitops-test/_git/gitops-test",
 | |
| 				referenceName:       "refs/heads/tags/v1.1",
 | |
| 				username:            "",
 | |
| 				password:            pat,
 | |
| 			},
 | |
| 			wantErr: false,
 | |
| 		},
 | |
| 	}
 | |
| 	for _, tt := range tests {
 | |
| 		t.Run(tt.name, func(t *testing.T) {
 | |
| 			dst := t.TempDir()
 | |
| 			repositoryUrl := fmt.Sprintf(tt.args.repositoryURLFormat, tt.args.password)
 | |
| 			err := service.CloneRepository(dst, repositoryUrl, tt.args.referenceName, "", "", false)
 | |
| 			assert.NoError(t, err)
 | |
| 			assert.FileExists(t, filepath.Join(dst, "README.md"))
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestService_ClonePrivateRepository_Azure(t *testing.T) {
 | |
| 	ensureIntegrationTest(t)
 | |
| 
 | |
| 	pat := getRequiredValue(t, "AZURE_DEVOPS_PAT")
 | |
| 	service := NewService(context.TODO())
 | |
| 
 | |
| 	dst := t.TempDir()
 | |
| 
 | |
| 	err := service.CloneRepository(dst, privateAzureRepoURL, "refs/heads/main", "", pat, false)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.FileExists(t, filepath.Join(dst, "README.md"))
 | |
| }
 | |
| 
 | |
| func TestService_LatestCommitID_Azure(t *testing.T) {
 | |
| 	ensureIntegrationTest(t)
 | |
| 
 | |
| 	pat := getRequiredValue(t, "AZURE_DEVOPS_PAT")
 | |
| 	service := NewService(context.TODO())
 | |
| 
 | |
| 	id, err := service.LatestCommitID(privateAzureRepoURL, "refs/heads/main", "", pat, false)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.NotEmpty(t, id, "cannot guarantee commit id, but it should be not empty")
 | |
| }
 | |
| 
 | |
| func TestService_ListRefs_Azure(t *testing.T) {
 | |
| 	ensureIntegrationTest(t)
 | |
| 
 | |
| 	accessToken := getRequiredValue(t, "AZURE_DEVOPS_PAT")
 | |
| 	username := getRequiredValue(t, "AZURE_DEVOPS_USERNAME")
 | |
| 	service := NewService(context.TODO())
 | |
| 
 | |
| 	refs, err := service.ListRefs(privateAzureRepoURL, username, accessToken, false, false)
 | |
| 	assert.NoError(t, err)
 | |
| 	assert.GreaterOrEqual(t, len(refs), 1)
 | |
| }
 | |
| 
 | |
| func TestService_ListRefs_Azure_Concurrently(t *testing.T) {
 | |
| 	ensureIntegrationTest(t)
 | |
| 
 | |
| 	accessToken := getRequiredValue(t, "AZURE_DEVOPS_PAT")
 | |
| 	username := getRequiredValue(t, "AZURE_DEVOPS_USERNAME")
 | |
| 	service := newService(context.TODO(), repositoryCacheSize, 200*time.Millisecond)
 | |
| 
 | |
| 	go service.ListRefs(privateAzureRepoURL, username, accessToken, false, false)
 | |
| 	service.ListRefs(privateAzureRepoURL, username, accessToken, false, false)
 | |
| 
 | |
| 	time.Sleep(2 * time.Second)
 | |
| }
 | |
| 
 | |
| func TestService_ListFiles_Azure(t *testing.T) {
 | |
| 	ensureIntegrationTest(t)
 | |
| 
 | |
| 	type expectResult struct {
 | |
| 		shouldFail   bool
 | |
| 		err          error
 | |
| 		matchedCount int
 | |
| 	}
 | |
| 	service := newService(context.TODO(), 0, 0)
 | |
| 	accessToken := getRequiredValue(t, "AZURE_DEVOPS_PAT")
 | |
| 	username := getRequiredValue(t, "AZURE_DEVOPS_USERNAME")
 | |
| 
 | |
| 	tests := []struct {
 | |
| 		name       string
 | |
| 		args       fetchOption
 | |
| 		extensions []string
 | |
| 		expect     expectResult
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "list tree with real repository and head ref but incorrect credential",
 | |
| 			args: fetchOption{
 | |
| 				baseOption: baseOption{
 | |
| 					repositoryUrl: privateAzureRepoURL,
 | |
| 					username:      "test-username",
 | |
| 					password:      "test-token",
 | |
| 				},
 | |
| 				referenceName: "refs/heads/main",
 | |
| 			},
 | |
| 			extensions: []string{},
 | |
| 			expect: expectResult{
 | |
| 				shouldFail: true,
 | |
| 				err:        gittypes.ErrAuthenticationFailure,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "list tree with real repository and head ref but no credential",
 | |
| 			args: fetchOption{
 | |
| 				baseOption: baseOption{
 | |
| 					repositoryUrl: privateAzureRepoURL,
 | |
| 					username:      "",
 | |
| 					password:      "",
 | |
| 				},
 | |
| 				referenceName: "refs/heads/main",
 | |
| 			},
 | |
| 			extensions: []string{},
 | |
| 			expect: expectResult{
 | |
| 				shouldFail: true,
 | |
| 				err:        gittypes.ErrAuthenticationFailure,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "list tree with real repository and head ref",
 | |
| 			args: fetchOption{
 | |
| 				baseOption: baseOption{
 | |
| 					repositoryUrl: privateAzureRepoURL,
 | |
| 					username:      username,
 | |
| 					password:      accessToken,
 | |
| 				},
 | |
| 				referenceName: "refs/heads/main",
 | |
| 			},
 | |
| 			extensions: []string{},
 | |
| 			expect: expectResult{
 | |
| 				err:          nil,
 | |
| 				matchedCount: 19,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "list tree with real repository and head ref and existing file extension",
 | |
| 			args: fetchOption{
 | |
| 				baseOption: baseOption{
 | |
| 					repositoryUrl: privateAzureRepoURL,
 | |
| 					username:      username,
 | |
| 					password:      accessToken,
 | |
| 				},
 | |
| 				referenceName: "refs/heads/main",
 | |
| 			},
 | |
| 			extensions: []string{"yml"},
 | |
| 			expect: expectResult{
 | |
| 				err:          nil,
 | |
| 				matchedCount: 2,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "list tree with real repository and head ref and non-existing file extension",
 | |
| 			args: fetchOption{
 | |
| 				baseOption: baseOption{
 | |
| 					repositoryUrl: privateAzureRepoURL,
 | |
| 					username:      username,
 | |
| 					password:      accessToken,
 | |
| 				},
 | |
| 				referenceName: "refs/heads/main",
 | |
| 			},
 | |
| 			extensions: []string{"hcl"},
 | |
| 			expect: expectResult{
 | |
| 				err:          nil,
 | |
| 				matchedCount: 2,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "list tree with real repository but non-existing ref",
 | |
| 			args: fetchOption{
 | |
| 				baseOption: baseOption{
 | |
| 					repositoryUrl: privateAzureRepoURL,
 | |
| 					username:      username,
 | |
| 					password:      accessToken,
 | |
| 				},
 | |
| 				referenceName: "refs/fake/feature",
 | |
| 			},
 | |
| 			extensions: []string{},
 | |
| 			expect: expectResult{
 | |
| 				shouldFail: true,
 | |
| 			},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "list tree with fake repository ",
 | |
| 			args: fetchOption{
 | |
| 				baseOption: baseOption{
 | |
| 					repositoryUrl: privateAzureRepoURL + "fake",
 | |
| 					username:      username,
 | |
| 					password:      accessToken,
 | |
| 				},
 | |
| 				referenceName: "refs/fake/feature",
 | |
| 			},
 | |
| 			extensions: []string{},
 | |
| 			expect: expectResult{
 | |
| 				shouldFail: true,
 | |
| 				err:        gittypes.ErrIncorrectRepositoryURL,
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, tt := range tests {
 | |
| 		t.Run(tt.name, func(t *testing.T) {
 | |
| 			paths, err := service.ListFiles(tt.args.repositoryUrl, tt.args.referenceName, tt.args.username, tt.args.password, false, false, tt.extensions, false)
 | |
| 			if tt.expect.shouldFail {
 | |
| 				assert.Error(t, err)
 | |
| 				if tt.expect.err != nil {
 | |
| 					assert.Equal(t, tt.expect.err, err)
 | |
| 				}
 | |
| 			} else {
 | |
| 				assert.NoError(t, err)
 | |
| 				if tt.expect.matchedCount > 0 {
 | |
| 					assert.Greater(t, len(paths), 0)
 | |
| 				}
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestService_ListFiles_Azure_Concurrently(t *testing.T) {
 | |
| 	ensureIntegrationTest(t)
 | |
| 
 | |
| 	accessToken := getRequiredValue(t, "AZURE_DEVOPS_PAT")
 | |
| 	username := getRequiredValue(t, "AZURE_DEVOPS_USERNAME")
 | |
| 	service := newService(context.TODO(), repositoryCacheSize, 200*time.Millisecond)
 | |
| 
 | |
| 	go service.ListFiles(privateAzureRepoURL, "refs/heads/main", username, accessToken, false, false, []string{}, false)
 | |
| 	service.ListFiles(privateAzureRepoURL, "refs/heads/main", username, accessToken, false, false, []string{}, false)
 | |
| 
 | |
| 	time.Sleep(2 * time.Second)
 | |
| }
 | |
| 
 | |
| func getRequiredValue(t *testing.T, name string) string {
 | |
| 	value, ok := os.LookupEnv(name)
 | |
| 	if !ok {
 | |
| 		t.Fatalf("can't find required env var \"%s\"", name)
 | |
| 	}
 | |
| 	return value
 | |
| }
 | |
| 
 | |
| func ensureIntegrationTest(t *testing.T) {
 | |
| 	if _, ok := os.LookupEnv("INTEGRATION_TEST"); !ok {
 | |
| 		t.Skip("skip an integration test")
 | |
| 	}
 | |
| }
 |