grafana/apps/provisioning/pkg/repository/factory_test.go

344 lines
10 KiB
Go

package repository
import (
"context"
"errors"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require"
provisioning "github.com/grafana/grafana/apps/provisioning/pkg/apis/provisioning/v0alpha1"
)
func TestNewFactory(t *testing.T) {
t.Run("creates factory with empty extras", func(t *testing.T) {
factory, err := ProvideFactory([]Extra{})
require.NoError(t, err)
require.NotNil(t, factory)
types := factory.Types()
assert.Empty(t, types)
})
t.Run("creates factory with multiple extras", func(t *testing.T) {
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
gitExtra := &MockExtra{}
gitExtra.On("Type").Return(provisioning.GitRepositoryType)
githubExtra := &MockExtra{}
githubExtra.On("Type").Return(provisioning.GitHubRepositoryType)
extras := []Extra{localExtra, gitExtra, githubExtra}
factory, err := ProvideFactory(extras)
require.NoError(t, err)
require.NotNil(t, factory)
types := factory.Types()
assert.Len(t, types, 3)
// Verify stable ordering - types should be sorted alphabetically
expectedTypes := []provisioning.RepositoryType{
provisioning.GitRepositoryType,
provisioning.GitHubRepositoryType,
provisioning.LocalRepositoryType,
}
assert.Equal(t, expectedTypes, types)
localExtra.AssertExpectations(t)
gitExtra.AssertExpectations(t)
githubExtra.AssertExpectations(t)
})
t.Run("returns error for duplicate repository types", func(t *testing.T) {
firstExtra := &MockExtra{}
firstExtra.On("Type").Return(provisioning.LocalRepositoryType)
secondExtra := &MockExtra{}
secondExtra.On("Type").Return(provisioning.LocalRepositoryType)
extras := []Extra{firstExtra, secondExtra}
factory, err := ProvideFactory(extras)
assert.Error(t, err)
assert.Nil(t, factory)
assert.Contains(t, err.Error(), "repository type \"local\" is already registered")
firstExtra.AssertExpectations(t)
secondExtra.AssertExpectations(t)
})
t.Run("returns error for duplicate among multiple different types", func(t *testing.T) {
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
gitExtra := &MockExtra{}
gitExtra.On("Type").Return(provisioning.GitRepositoryType)
duplicateGitExtra := &MockExtra{}
duplicateGitExtra.On("Type").Return(provisioning.GitRepositoryType)
extras := []Extra{localExtra, gitExtra, duplicateGitExtra}
factory, err := ProvideFactory(extras)
assert.Error(t, err)
assert.Nil(t, factory)
assert.Contains(t, err.Error(), "repository type \"git\" is already registered")
localExtra.AssertExpectations(t)
gitExtra.AssertExpectations(t)
duplicateGitExtra.AssertExpectations(t)
})
t.Run("handles nil extras slice", func(t *testing.T) {
factory, err := ProvideFactory(nil)
require.NoError(t, err)
require.NotNil(t, factory)
types := factory.Types()
assert.Empty(t, types)
})
}
func TestFactory_Types(t *testing.T) {
t.Run("returns empty slice for factory with no extras", func(t *testing.T) {
factory, err := ProvideFactory([]Extra{})
require.NoError(t, err)
types := factory.Types()
assert.Empty(t, types)
})
t.Run("returns all registered repository types in stable order", func(t *testing.T) {
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
gitExtra := &MockExtra{}
gitExtra.On("Type").Return(provisioning.GitRepositoryType)
githubExtra := &MockExtra{}
githubExtra.On("Type").Return(provisioning.GitHubRepositoryType)
bitbucketExtra := &MockExtra{}
bitbucketExtra.On("Type").Return(provisioning.BitbucketRepositoryType)
gitlabExtra := &MockExtra{}
gitlabExtra.On("Type").Return(provisioning.GitLabRepositoryType)
extras := []Extra{localExtra, gitExtra, githubExtra, bitbucketExtra, gitlabExtra}
factory, err := ProvideFactory(extras)
require.NoError(t, err)
types := factory.Types()
assert.Len(t, types, 5)
// Verify stable ordering - types should be sorted alphabetically
expectedTypes := []provisioning.RepositoryType{
provisioning.BitbucketRepositoryType,
provisioning.GitRepositoryType,
provisioning.GitHubRepositoryType,
provisioning.GitLabRepositoryType,
provisioning.LocalRepositoryType,
}
assert.Equal(t, expectedTypes, types)
localExtra.AssertExpectations(t)
gitExtra.AssertExpectations(t)
githubExtra.AssertExpectations(t)
bitbucketExtra.AssertExpectations(t)
gitlabExtra.AssertExpectations(t)
})
t.Run("returns consistent order across multiple calls", func(t *testing.T) {
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
gitExtra := &MockExtra{}
gitExtra.On("Type").Return(provisioning.GitRepositoryType)
githubExtra := &MockExtra{}
githubExtra.On("Type").Return(provisioning.GitHubRepositoryType)
extras := []Extra{githubExtra, localExtra, gitExtra} // Intentionally unordered
factory, err := ProvideFactory(extras)
require.NoError(t, err)
types1 := factory.Types()
types2 := factory.Types()
types3 := factory.Types()
// All calls should return the same order
assert.Equal(t, types1, types2)
assert.Equal(t, types2, types3)
// Verify the order is alphabetical
expectedTypes := []provisioning.RepositoryType{
provisioning.GitRepositoryType,
provisioning.GitHubRepositoryType,
provisioning.LocalRepositoryType,
}
assert.Equal(t, expectedTypes, types1)
localExtra.AssertExpectations(t)
gitExtra.AssertExpectations(t)
githubExtra.AssertExpectations(t)
})
}
func TestFactory_Build(t *testing.T) {
t.Run("successfully builds repository with matching type", func(t *testing.T) {
expectedRepo := &MockConfigRepository{}
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
localExtra.On("Build", mock.Anything, mock.Anything).Return(expectedRepo, nil)
factory, err := ProvideFactory([]Extra{localExtra})
require.NoError(t, err)
ctx := context.Background()
repoConfig := &provisioning.Repository{
Spec: provisioning.RepositorySpec{
Type: provisioning.LocalRepositoryType,
},
}
result, err := factory.Build(ctx, repoConfig)
require.NoError(t, err)
assert.Equal(t, expectedRepo, result)
localExtra.AssertExpectations(t)
})
t.Run("returns error for unsupported repository type", func(t *testing.T) {
gitExtra := &MockExtra{}
gitExtra.On("Type").Return(provisioning.GitRepositoryType)
factory, err := ProvideFactory([]Extra{gitExtra})
require.NoError(t, err)
ctx := context.Background()
repoConfig := &provisioning.Repository{
Spec: provisioning.RepositorySpec{
Type: provisioning.LocalRepositoryType, // Different from registered type
},
}
result, err := factory.Build(ctx, repoConfig)
assert.Error(t, err)
assert.Nil(t, result)
assert.Contains(t, err.Error(), "repository type \"local\" is not supported")
gitExtra.AssertNotCalled(t, "Build")
gitExtra.AssertExpectations(t)
})
t.Run("propagates error from extra.Build", func(t *testing.T) {
expectedError := errors.New("build failed")
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
localExtra.On("Build", mock.Anything, mock.Anything).Return(nil, expectedError)
factory, err := ProvideFactory([]Extra{localExtra})
require.NoError(t, err)
ctx := context.Background()
repoConfig := &provisioning.Repository{
Spec: provisioning.RepositorySpec{
Type: provisioning.LocalRepositoryType,
},
}
result, err := factory.Build(ctx, repoConfig)
assert.Error(t, err)
assert.Equal(t, expectedError, err)
assert.Nil(t, result)
localExtra.AssertExpectations(t)
})
t.Run("finds correct extra among multiple", func(t *testing.T) {
gitRepo := &MockConfigRepository{}
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
gitExtra := &MockExtra{}
gitExtra.On("Type").Return(provisioning.GitRepositoryType)
gitExtra.On("Build", mock.Anything, mock.Anything).Return(gitRepo, nil)
factory, err := ProvideFactory([]Extra{localExtra, gitExtra})
require.NoError(t, err)
ctx := context.Background()
repoConfig := &provisioning.Repository{
Spec: provisioning.RepositorySpec{
Type: provisioning.GitRepositoryType,
},
}
result, err := factory.Build(ctx, repoConfig)
require.NoError(t, err)
assert.Equal(t, gitRepo, result)
localExtra.AssertNotCalled(t, "Build") // Should not be called
gitExtra.AssertExpectations(t) // Should be called
localExtra.AssertExpectations(t)
})
t.Run("handles empty repository type", func(t *testing.T) {
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
factory, err := ProvideFactory([]Extra{localExtra})
require.NoError(t, err)
ctx := context.Background()
repoConfig := &provisioning.Repository{
Spec: provisioning.RepositorySpec{
Type: "", // Empty type
},
}
result, err := factory.Build(ctx, repoConfig)
assert.Error(t, err)
assert.Nil(t, result)
assert.Contains(t, err.Error(), "repository type \"\" is not supported")
localExtra.AssertNotCalled(t, "Build")
localExtra.AssertExpectations(t)
})
t.Run("passes context correctly to extra.Build", func(t *testing.T) {
localRepo := &MockConfigRepository{}
localExtra := &MockExtra{}
localExtra.On("Type").Return(provisioning.LocalRepositoryType)
// Create context with value to verify it's passed through
type testKey string
ctx := context.WithValue(context.Background(), testKey("test"), "value")
// Use a custom matcher to verify the context is passed correctly
localExtra.On("Build", mock.MatchedBy(func(c context.Context) bool {
return c.Value(testKey("test")) == "value"
}), mock.Anything).Return(localRepo, nil)
factory, err := ProvideFactory([]Extra{localExtra})
require.NoError(t, err)
repoConfig := &provisioning.Repository{
Spec: provisioning.RepositorySpec{
Type: provisioning.LocalRepositoryType,
},
}
_, err = factory.Build(ctx, repoConfig)
require.NoError(t, err)
localExtra.AssertExpectations(t)
})
}