mirror of https://github.com/grafana/grafana.git
				
				
				
			
		
			
				
	
	
		
			1256 lines
		
	
	
		
			52 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			1256 lines
		
	
	
		
			52 KiB
		
	
	
	
		
			Go
		
	
	
	
| package folderimpl
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"math/rand"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| 	"github.com/stretchr/testify/mock"
 | |
| 	"github.com/stretchr/testify/require"
 | |
| 
 | |
| 	"github.com/grafana/grafana/pkg/api/routing"
 | |
| 	"github.com/grafana/grafana/pkg/bus"
 | |
| 	"github.com/grafana/grafana/pkg/infra/db"
 | |
| 	"github.com/grafana/grafana/pkg/infra/db/dbtest"
 | |
| 	"github.com/grafana/grafana/pkg/infra/log"
 | |
| 	"github.com/grafana/grafana/pkg/infra/tracing"
 | |
| 	"github.com/grafana/grafana/pkg/services/accesscontrol"
 | |
| 	"github.com/grafana/grafana/pkg/services/accesscontrol/acimpl"
 | |
| 	"github.com/grafana/grafana/pkg/services/accesscontrol/actest"
 | |
| 	acmock "github.com/grafana/grafana/pkg/services/accesscontrol/mock"
 | |
| 	"github.com/grafana/grafana/pkg/services/dashboards"
 | |
| 	"github.com/grafana/grafana/pkg/services/dashboards/database"
 | |
| 	"github.com/grafana/grafana/pkg/services/dashboards/service"
 | |
| 	"github.com/grafana/grafana/pkg/services/featuremgmt"
 | |
| 	"github.com/grafana/grafana/pkg/services/folder"
 | |
| 	"github.com/grafana/grafana/pkg/services/folder/foldertest"
 | |
| 	"github.com/grafana/grafana/pkg/services/guardian"
 | |
| 	"github.com/grafana/grafana/pkg/services/libraryelements"
 | |
| 	"github.com/grafana/grafana/pkg/services/libraryelements/model"
 | |
| 	"github.com/grafana/grafana/pkg/services/librarypanels"
 | |
| 	"github.com/grafana/grafana/pkg/services/ngalert/models"
 | |
| 	ngstore "github.com/grafana/grafana/pkg/services/ngalert/store"
 | |
| 	"github.com/grafana/grafana/pkg/services/quota/quotatest"
 | |
| 	"github.com/grafana/grafana/pkg/services/sqlstore"
 | |
| 	"github.com/grafana/grafana/pkg/services/store/entity"
 | |
| 	"github.com/grafana/grafana/pkg/services/tag/tagimpl"
 | |
| 	"github.com/grafana/grafana/pkg/services/user"
 | |
| 	"github.com/grafana/grafana/pkg/setting"
 | |
| 	"github.com/grafana/grafana/pkg/util"
 | |
| )
 | |
| 
 | |
| var orgID = int64(1)
 | |
| var usr = &user.SignedInUser{UserID: 1, OrgID: orgID}
 | |
| 
 | |
| func TestIntegrationProvideFolderService(t *testing.T) {
 | |
| 	if testing.Short() {
 | |
| 		t.Skip("skipping integration test")
 | |
| 	}
 | |
| 	t.Run("should register scope resolvers", func(t *testing.T) {
 | |
| 		cfg := setting.NewCfg()
 | |
| 		ac := acmock.New()
 | |
| 		db := sqlstore.InitTestDB(t)
 | |
| 		ProvideService(ac, bus.ProvideBus(tracing.InitializeTracerForTest()), cfg, nil, nil, db, &featuremgmt.FeatureManager{})
 | |
| 
 | |
| 		require.Len(t, ac.Calls.RegisterAttributeScopeResolver, 3)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestIntegrationFolderService(t *testing.T) {
 | |
| 	if testing.Short() {
 | |
| 		t.Skip("skipping integration test")
 | |
| 	}
 | |
| 	t.Run("Folder service tests", func(t *testing.T) {
 | |
| 		dashStore := &dashboards.FakeDashboardStore{}
 | |
| 		db := sqlstore.InitTestDB(t)
 | |
| 		nestedFolderStore := ProvideStore(db, db.Cfg, featuremgmt.WithFeatures([]interface{}{"nestedFolders"}))
 | |
| 
 | |
| 		folderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 		cfg := setting.NewCfg()
 | |
| 		features := featuremgmt.WithFeatures()
 | |
| 
 | |
| 		service := &Service{
 | |
| 			cfg:                  cfg,
 | |
| 			log:                  log.New("test-folder-service"),
 | |
| 			dashboardStore:       dashStore,
 | |
| 			dashboardFolderStore: folderStore,
 | |
| 			store:                nestedFolderStore,
 | |
| 			features:             features,
 | |
| 			bus:                  bus.ProvideBus(tracing.InitializeTracerForTest()),
 | |
| 			db:                   db,
 | |
| 			accessControl:        acimpl.ProvideAccessControl(cfg),
 | |
| 		}
 | |
| 
 | |
| 		t.Run("Given user has no permissions", func(t *testing.T) {
 | |
| 			origNewGuardian := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{})
 | |
| 
 | |
| 			folderId := rand.Int63()
 | |
| 			folderUID := util.GenerateShortUID()
 | |
| 
 | |
| 			f := folder.NewFolder("Folder", "")
 | |
| 			f.ID = folderId
 | |
| 			f.UID = folderUID
 | |
| 
 | |
| 			folderStore.On("GetFolderByID", mock.Anything, orgID, folderId).Return(f, nil)
 | |
| 			folderStore.On("GetFolderByUID", mock.Anything, orgID, folderUID).Return(f, nil)
 | |
| 
 | |
| 			t.Run("When get folder by id should return access denied error", func(t *testing.T) {
 | |
| 				_, err := service.Get(context.Background(), &folder.GetFolderQuery{
 | |
| 					ID:           &folderId,
 | |
| 					OrgID:        orgID,
 | |
| 					SignedInUser: usr,
 | |
| 				})
 | |
| 				require.Equal(t, err, dashboards.ErrFolderAccessDenied)
 | |
| 			})
 | |
| 
 | |
| 			var zeroInt int64 = 0
 | |
| 			t.Run("When get folder by id, with id = 0 should return default folder", func(t *testing.T) {
 | |
| 				foldr, err := service.Get(context.Background(), &folder.GetFolderQuery{
 | |
| 					ID:           &zeroInt,
 | |
| 					OrgID:        orgID,
 | |
| 					SignedInUser: usr,
 | |
| 				})
 | |
| 				require.NoError(t, err)
 | |
| 				require.Equal(t, foldr, &folder.Folder{ID: 0, Title: "General"})
 | |
| 			})
 | |
| 
 | |
| 			t.Run("When get folder by uid should return access denied error", func(t *testing.T) {
 | |
| 				_, err := service.Get(context.Background(), &folder.GetFolderQuery{
 | |
| 					UID:          &folderUID,
 | |
| 					OrgID:        orgID,
 | |
| 					SignedInUser: usr,
 | |
| 				})
 | |
| 				require.Equal(t, err, dashboards.ErrFolderAccessDenied)
 | |
| 			})
 | |
| 
 | |
| 			t.Run("When creating folder should return access denied error", func(t *testing.T) {
 | |
| 				dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil).Times(2)
 | |
| 				_, err := service.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 					OrgID:        orgID,
 | |
| 					Title:        f.Title,
 | |
| 					UID:          folderUID,
 | |
| 					SignedInUser: usr,
 | |
| 				})
 | |
| 				require.Equal(t, err, dashboards.ErrFolderAccessDenied)
 | |
| 			})
 | |
| 
 | |
| 			title := "Folder-TEST"
 | |
| 			t.Run("When updating folder should return access denied error", func(t *testing.T) {
 | |
| 				folderResult := dashboards.NewDashboard("dashboard-test")
 | |
| 				folderResult.IsFolder = true
 | |
| 				dashStore.On("GetDashboard", mock.Anything, mock.AnythingOfType("*dashboards.GetDashboardQuery")).Return(folderResult, nil)
 | |
| 				_, err := service.Update(context.Background(), &folder.UpdateFolderCommand{
 | |
| 					UID:          folderUID,
 | |
| 					OrgID:        orgID,
 | |
| 					NewTitle:     &title,
 | |
| 					SignedInUser: usr,
 | |
| 				})
 | |
| 				require.Equal(t, err, dashboards.ErrFolderAccessDenied)
 | |
| 			})
 | |
| 
 | |
| 			t.Run("When deleting folder by uid should return access denied error", func(t *testing.T) {
 | |
| 				newFolder := folder.NewFolder("Folder", "")
 | |
| 				newFolder.UID = folderUID
 | |
| 
 | |
| 				folderStore.On("GetFolderByID", mock.Anything, orgID, folderId).Return(newFolder, nil)
 | |
| 				folderStore.On("GetFolderByUID", mock.Anything, orgID, folderUID).Return(newFolder, nil)
 | |
| 
 | |
| 				err := service.Delete(context.Background(), &folder.DeleteFolderCommand{
 | |
| 					UID:              folderUID,
 | |
| 					OrgID:            orgID,
 | |
| 					ForceDeleteRules: false,
 | |
| 					SignedInUser:     usr,
 | |
| 				})
 | |
| 				require.Error(t, err)
 | |
| 				require.Equal(t, err, dashboards.ErrFolderAccessDenied)
 | |
| 			})
 | |
| 
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = origNewGuardian
 | |
| 			})
 | |
| 		})
 | |
| 
 | |
| 		t.Run("Given user has permission to save", func(t *testing.T) {
 | |
| 			origNewGuardian := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			service.features = featuremgmt.WithFeatures()
 | |
| 
 | |
| 			t.Run("When creating folder should not return access denied error", func(t *testing.T) {
 | |
| 				dash := dashboards.NewDashboardFolder("Test-Folder")
 | |
| 				dash.ID = rand.Int63()
 | |
| 				f := dashboards.FromDashboard(dash)
 | |
| 
 | |
| 				dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 				dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(dash, nil).Once()
 | |
| 				folderStore.On("GetFolderByID", mock.Anything, orgID, dash.ID).Return(f, nil)
 | |
| 
 | |
| 				actualFolder, err := service.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 					OrgID:        orgID,
 | |
| 					Title:        dash.Title,
 | |
| 					UID:          "someuid",
 | |
| 					SignedInUser: usr,
 | |
| 				})
 | |
| 				require.NoError(t, err)
 | |
| 				require.Equal(t, f, actualFolder)
 | |
| 			})
 | |
| 
 | |
| 			t.Run("When creating folder should return error if uid is general", func(t *testing.T) {
 | |
| 				dash := dashboards.NewDashboardFolder("Test-Folder")
 | |
| 				dash.ID = rand.Int63()
 | |
| 
 | |
| 				_, err := service.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 					OrgID:        orgID,
 | |
| 					Title:        dash.Title,
 | |
| 					UID:          "general",
 | |
| 					SignedInUser: usr,
 | |
| 				})
 | |
| 				require.ErrorIs(t, err, dashboards.ErrFolderInvalidUID)
 | |
| 			})
 | |
| 
 | |
| 			t.Run("When updating folder should not return access denied error", func(t *testing.T) {
 | |
| 				dashboardFolder := dashboards.NewDashboardFolder("Folder")
 | |
| 				dashboardFolder.ID = rand.Int63()
 | |
| 				dashboardFolder.UID = util.GenerateShortUID()
 | |
| 				f := dashboards.FromDashboard(dashboardFolder)
 | |
| 
 | |
| 				dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 				dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(dashboardFolder, nil)
 | |
| 				folderStore.On("GetFolderByID", mock.Anything, orgID, dashboardFolder.ID).Return(f, nil)
 | |
| 
 | |
| 				title := "TEST-Folder"
 | |
| 				req := &folder.UpdateFolderCommand{
 | |
| 					UID:          dashboardFolder.UID,
 | |
| 					OrgID:        orgID,
 | |
| 					NewTitle:     &title,
 | |
| 					SignedInUser: usr,
 | |
| 				}
 | |
| 
 | |
| 				reqResult, err := service.Update(context.Background(), req)
 | |
| 				require.NoError(t, err)
 | |
| 				require.Equal(t, f, reqResult)
 | |
| 			})
 | |
| 
 | |
| 			t.Run("When deleting folder by uid should not return access denied error", func(t *testing.T) {
 | |
| 				f := folder.NewFolder(util.GenerateShortUID(), "")
 | |
| 				f.ID = rand.Int63()
 | |
| 				f.UID = util.GenerateShortUID()
 | |
| 				folderStore.On("GetFolders", mock.Anything, orgID, []string{f.UID}).Return(map[string]*folder.Folder{f.UID: f}, nil)
 | |
| 
 | |
| 				var actualCmd *dashboards.DeleteDashboardCommand
 | |
| 				dashStore.On("DeleteDashboard", mock.Anything, mock.Anything).Run(func(args mock.Arguments) {
 | |
| 					actualCmd = args.Get(1).(*dashboards.DeleteDashboardCommand)
 | |
| 				}).Return(nil).Once()
 | |
| 
 | |
| 				expectedForceDeleteRules := rand.Int63()%2 == 0
 | |
| 				err := service.Delete(context.Background(), &folder.DeleteFolderCommand{
 | |
| 					UID:              f.UID,
 | |
| 					OrgID:            orgID,
 | |
| 					ForceDeleteRules: expectedForceDeleteRules,
 | |
| 					SignedInUser:     usr,
 | |
| 				})
 | |
| 				require.NoError(t, err)
 | |
| 				require.NotNil(t, actualCmd)
 | |
| 				require.Equal(t, f.ID, actualCmd.ID)
 | |
| 				require.Equal(t, orgID, actualCmd.OrgID)
 | |
| 				require.Equal(t, expectedForceDeleteRules, actualCmd.ForceDeleteFolderRules)
 | |
| 			})
 | |
| 
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = origNewGuardian
 | |
| 			})
 | |
| 		})
 | |
| 
 | |
| 		t.Run("Given user has permission to view", func(t *testing.T) {
 | |
| 			origNewGuardian := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanViewValue: true})
 | |
| 
 | |
| 			t.Run("When get folder by id should return folder", func(t *testing.T) {
 | |
| 				expected := folder.NewFolder(util.GenerateShortUID(), "")
 | |
| 				expected.ID = rand.Int63()
 | |
| 
 | |
| 				folderStore.On("GetFolderByID", mock.Anything, orgID, expected.ID).Return(expected, nil)
 | |
| 
 | |
| 				actual, err := service.getFolderByID(context.Background(), expected.ID, orgID)
 | |
| 				require.Equal(t, expected, actual)
 | |
| 				require.NoError(t, err)
 | |
| 			})
 | |
| 
 | |
| 			t.Run("When get folder by uid should return folder", func(t *testing.T) {
 | |
| 				expected := folder.NewFolder(util.GenerateShortUID(), "")
 | |
| 				expected.UID = util.GenerateShortUID()
 | |
| 
 | |
| 				folderStore.On("GetFolderByUID", mock.Anything, orgID, expected.UID).Return(expected, nil)
 | |
| 
 | |
| 				actual, err := service.getFolderByUID(context.Background(), orgID, expected.UID)
 | |
| 				require.Equal(t, expected, actual)
 | |
| 				require.NoError(t, err)
 | |
| 			})
 | |
| 
 | |
| 			t.Run("When get folder by title should return folder", func(t *testing.T) {
 | |
| 				expected := folder.NewFolder("TEST-"+util.GenerateShortUID(), "")
 | |
| 
 | |
| 				folderStore.On("GetFolderByTitle", mock.Anything, orgID, expected.Title).Return(expected, nil)
 | |
| 
 | |
| 				actual, err := service.getFolderByTitle(context.Background(), orgID, expected.Title)
 | |
| 				require.Equal(t, expected, actual)
 | |
| 				require.NoError(t, err)
 | |
| 			})
 | |
| 
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = origNewGuardian
 | |
| 			})
 | |
| 		})
 | |
| 
 | |
| 		t.Run("Should map errors correct", func(t *testing.T) {
 | |
| 			testCases := []struct {
 | |
| 				ActualError   error
 | |
| 				ExpectedError error
 | |
| 			}{
 | |
| 				{ActualError: dashboards.ErrDashboardTitleEmpty, ExpectedError: dashboards.ErrFolderTitleEmpty},
 | |
| 				{ActualError: dashboards.ErrDashboardUpdateAccessDenied, ExpectedError: dashboards.ErrFolderAccessDenied},
 | |
| 				{ActualError: dashboards.ErrDashboardWithSameNameInFolderExists, ExpectedError: dashboards.ErrFolderSameNameExists},
 | |
| 				{ActualError: dashboards.ErrDashboardWithSameUIDExists, ExpectedError: dashboards.ErrFolderWithSameUIDExists},
 | |
| 				{ActualError: dashboards.ErrDashboardVersionMismatch, ExpectedError: dashboards.ErrFolderVersionMismatch},
 | |
| 				{ActualError: dashboards.ErrDashboardNotFound, ExpectedError: dashboards.ErrFolderNotFound},
 | |
| 				{ActualError: dashboards.ErrDashboardInvalidUid, ExpectedError: dashboards.ErrDashboardInvalidUid},
 | |
| 			}
 | |
| 
 | |
| 			for _, tc := range testCases {
 | |
| 				actualError := toFolderError(tc.ActualError)
 | |
| 				assert.EqualErrorf(t, actualError, tc.ExpectedError.Error(),
 | |
| 					"For error '%s' expected error '%s', actual '%s'", tc.ActualError, tc.ExpectedError, actualError)
 | |
| 			}
 | |
| 		})
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestIntegrationNestedFolderService(t *testing.T) {
 | |
| 	if testing.Short() {
 | |
| 		t.Skip("skipping integration test")
 | |
| 	}
 | |
| 	db := sqlstore.InitTestDB(t)
 | |
| 	quotaService := quotatest.New(false, nil)
 | |
| 	folderStore := ProvideDashboardFolderStore(db)
 | |
| 
 | |
| 	cfg := setting.NewCfg()
 | |
| 
 | |
| 	featuresFlagOn := featuremgmt.WithFeatures("nestedFolders")
 | |
| 	dashStore, err := database.ProvideDashboardStore(db, db.Cfg, featuresFlagOn, tagimpl.ProvideService(db, db.Cfg), quotaService)
 | |
| 	require.NoError(t, err)
 | |
| 	nestedFolderStore := ProvideStore(db, db.Cfg, featuresFlagOn)
 | |
| 
 | |
| 	b := bus.ProvideBus(tracing.InitializeTracerForTest())
 | |
| 	ac := acimpl.ProvideAccessControl(cfg)
 | |
| 
 | |
| 	serviceWithFlagOn := &Service{
 | |
| 		cfg:                  cfg,
 | |
| 		log:                  log.New("test-folder-service"),
 | |
| 		dashboardStore:       dashStore,
 | |
| 		dashboardFolderStore: folderStore,
 | |
| 		store:                nestedFolderStore,
 | |
| 		features:             featuresFlagOn,
 | |
| 		bus:                  b,
 | |
| 		db:                   db,
 | |
| 		accessControl:        ac,
 | |
| 		registry:             make(map[string]folder.RegistryService),
 | |
| 	}
 | |
| 
 | |
| 	signedInUser := user.SignedInUser{UserID: 1, OrgID: orgID, Permissions: map[int64]map[string][]string{
 | |
| 		orgID: {
 | |
| 			dashboards.ActionFoldersCreate: {},
 | |
| 			dashboards.ActionFoldersWrite:  {dashboards.ScopeFoldersAll}},
 | |
| 	}}
 | |
| 	createCmd := folder.CreateFolderCommand{
 | |
| 		OrgID:        orgID,
 | |
| 		ParentUID:    "",
 | |
| 		SignedInUser: &signedInUser,
 | |
| 	}
 | |
| 
 | |
| 	libraryElementCmd := model.CreateLibraryElementCommand{
 | |
| 		Model: []byte(`
 | |
| 		{
 | |
| 		  "datasource": "${DS_GDEV-TESTDATA}",
 | |
| 		  "id": 1,
 | |
| 		  "title": "Text - Library Panel",
 | |
| 		  "type": "text",
 | |
| 		  "description": "A description"
 | |
| 		}
 | |
| 	`),
 | |
| 		Kind: int64(model.PanelElement),
 | |
| 	}
 | |
| 	routeRegister := routing.NewRouteRegister()
 | |
| 
 | |
| 	folderPermissions := acmock.NewMockedPermissionsService()
 | |
| 	dashboardPermissions := acmock.NewMockedPermissionsService()
 | |
| 
 | |
| 	t.Run("Should get descendant counts", func(t *testing.T) {
 | |
| 		depth := 5
 | |
| 		t.Run("With nested folder feature flag on", func(t *testing.T) {
 | |
| 			origNewGuardian := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{
 | |
| 				CanSaveValue: true,
 | |
| 				CanViewValue: true,
 | |
| 				// CanEditValue is required to create library elements
 | |
| 				CanEditValue: true,
 | |
| 			})
 | |
| 
 | |
| 			dashSrv, err := service.ProvideDashboardServiceImpl(cfg, dashStore, folderStore, nil, featuresFlagOn, folderPermissions, dashboardPermissions, ac, serviceWithFlagOn)
 | |
| 			require.NoError(t, err)
 | |
| 
 | |
| 			alertStore, err := ngstore.ProvideDBStore(cfg, featuresFlagOn, db, serviceWithFlagOn, ac, dashSrv)
 | |
| 			require.NoError(t, err)
 | |
| 
 | |
| 			elementService := libraryelements.ProvideService(cfg, db, routeRegister, serviceWithFlagOn, featuresFlagOn)
 | |
| 			lps, err := librarypanels.ProvideService(cfg, db, routeRegister, elementService, serviceWithFlagOn)
 | |
| 			require.NoError(t, err)
 | |
| 
 | |
| 			ancestorUIDs := CreateSubtreeInStore(t, nestedFolderStore, serviceWithFlagOn, depth, "getDescendantCountsOn", createCmd)
 | |
| 
 | |
| 			parent, err := serviceWithFlagOn.dashboardFolderStore.GetFolderByUID(context.Background(), orgID, ancestorUIDs[0])
 | |
| 			require.NoError(t, err)
 | |
| 			subfolder, err := serviceWithFlagOn.dashboardFolderStore.GetFolderByUID(context.Background(), orgID, ancestorUIDs[1])
 | |
| 			require.NoError(t, err)
 | |
| 			_ = insertTestDashboard(t, serviceWithFlagOn.dashboardStore, "dashboard in parent", orgID, parent.ID, "prod")
 | |
| 			_ = insertTestDashboard(t, serviceWithFlagOn.dashboardStore, "dashboard in subfolder", orgID, subfolder.ID, "prod")
 | |
| 			_ = createRule(t, alertStore, parent.UID, "parent alert")
 | |
| 			_ = createRule(t, alertStore, subfolder.UID, "sub alert")
 | |
| 
 | |
| 			libraryElementCmd.FolderID = parent.ID
 | |
| 			_, err = lps.LibraryElementService.CreateElement(context.Background(), &signedInUser, libraryElementCmd)
 | |
| 			require.NoError(t, err)
 | |
| 			libraryElementCmd.FolderID = subfolder.ID
 | |
| 			_, err = lps.LibraryElementService.CreateElement(context.Background(), &signedInUser, libraryElementCmd)
 | |
| 			require.NoError(t, err)
 | |
| 
 | |
| 			countCmd := folder.GetDescendantCountsQuery{
 | |
| 				UID:          &ancestorUIDs[0],
 | |
| 				OrgID:        orgID,
 | |
| 				SignedInUser: &signedInUser,
 | |
| 			}
 | |
| 			m, err := serviceWithFlagOn.GetDescendantCounts(context.Background(), &countCmd)
 | |
| 			require.NoError(t, err)
 | |
| 			require.Equal(t, int64(depth-1), m[entity.StandardKindFolder])
 | |
| 			require.Equal(t, int64(2), m[entity.StandardKindDashboard])
 | |
| 			require.Equal(t, int64(2), m[entity.StandardKindAlertRule])
 | |
| 			require.Equal(t, int64(2), m[entity.StandardKindLibraryPanel])
 | |
| 
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = origNewGuardian
 | |
| 				for _, uid := range ancestorUIDs {
 | |
| 					err := serviceWithFlagOn.store.Delete(context.Background(), uid, orgID)
 | |
| 					assert.NoError(t, err)
 | |
| 				}
 | |
| 			})
 | |
| 		})
 | |
| 		t.Run("With nested folder feature flag off", func(t *testing.T) {
 | |
| 			featuresFlagOff := featuremgmt.WithFeatures()
 | |
| 			dashStore, err := database.ProvideDashboardStore(db, db.Cfg, featuresFlagOff, tagimpl.ProvideService(db, db.Cfg), quotaService)
 | |
| 			require.NoError(t, err)
 | |
| 			nestedFolderStore := ProvideStore(db, db.Cfg, featuresFlagOff)
 | |
| 
 | |
| 			serviceWithFlagOff := &Service{
 | |
| 				cfg:                  cfg,
 | |
| 				log:                  log.New("test-folder-service"),
 | |
| 				dashboardStore:       dashStore,
 | |
| 				dashboardFolderStore: folderStore,
 | |
| 				store:                nestedFolderStore,
 | |
| 				features:             featuresFlagOff,
 | |
| 				bus:                  b,
 | |
| 				db:                   db,
 | |
| 				registry:             make(map[string]folder.RegistryService),
 | |
| 			}
 | |
| 
 | |
| 			origNewGuardian := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{
 | |
| 				CanSaveValue: true,
 | |
| 				CanViewValue: true,
 | |
| 				// CanEditValue is required to create library elements
 | |
| 				CanEditValue: true,
 | |
| 			})
 | |
| 
 | |
| 			dashSrv, err := service.ProvideDashboardServiceImpl(cfg, dashStore, folderStore, nil, featuresFlagOff,
 | |
| 				folderPermissions, dashboardPermissions, ac, serviceWithFlagOff)
 | |
| 			require.NoError(t, err)
 | |
| 
 | |
| 			alertStore, err := ngstore.ProvideDBStore(cfg, featuresFlagOff, db, serviceWithFlagOff, ac, dashSrv)
 | |
| 			require.NoError(t, err)
 | |
| 
 | |
| 			elementService := libraryelements.ProvideService(cfg, db, routeRegister, serviceWithFlagOff, featuresFlagOff)
 | |
| 			lps, err := librarypanels.ProvideService(cfg, db, routeRegister, elementService, serviceWithFlagOff)
 | |
| 			require.NoError(t, err)
 | |
| 
 | |
| 			ancestorUIDs := CreateSubtreeInStore(t, nestedFolderStore, serviceWithFlagOn, depth, "getDescendantCountsOff", createCmd)
 | |
| 
 | |
| 			parent, err := serviceWithFlagOn.dashboardFolderStore.GetFolderByUID(context.Background(), orgID, ancestorUIDs[0])
 | |
| 			require.NoError(t, err)
 | |
| 			subfolder, err := serviceWithFlagOn.dashboardFolderStore.GetFolderByUID(context.Background(), orgID, ancestorUIDs[1])
 | |
| 			require.NoError(t, err)
 | |
| 			_ = insertTestDashboard(t, serviceWithFlagOn.dashboardStore, "dashboard in parent", orgID, parent.ID, "prod")
 | |
| 			_ = insertTestDashboard(t, serviceWithFlagOn.dashboardStore, "dashboard in subfolder", orgID, subfolder.ID, "prod")
 | |
| 			_ = createRule(t, alertStore, parent.UID, "parent alert")
 | |
| 			_ = createRule(t, alertStore, subfolder.UID, "sub alert")
 | |
| 
 | |
| 			libraryElementCmd.FolderID = parent.ID
 | |
| 			_, err = lps.LibraryElementService.CreateElement(context.Background(), &signedInUser, libraryElementCmd)
 | |
| 			require.NoError(t, err)
 | |
| 			libraryElementCmd.FolderID = subfolder.ID
 | |
| 			_, err = lps.LibraryElementService.CreateElement(context.Background(), &signedInUser, libraryElementCmd)
 | |
| 			require.NoError(t, err)
 | |
| 
 | |
| 			countCmd := folder.GetDescendantCountsQuery{
 | |
| 				UID:          &ancestorUIDs[0],
 | |
| 				OrgID:        orgID,
 | |
| 				SignedInUser: &signedInUser,
 | |
| 			}
 | |
| 			m, err := serviceWithFlagOff.GetDescendantCounts(context.Background(), &countCmd)
 | |
| 			require.NoError(t, err)
 | |
| 			require.Equal(t, int64(0), m[entity.StandardKindFolder])
 | |
| 			require.Equal(t, int64(1), m[entity.StandardKindDashboard])
 | |
| 			require.Equal(t, int64(1), m[entity.StandardKindAlertRule])
 | |
| 			require.Equal(t, int64(1), m[entity.StandardKindLibraryPanel])
 | |
| 
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = origNewGuardian
 | |
| 				for _, uid := range ancestorUIDs {
 | |
| 					err := serviceWithFlagOn.store.Delete(context.Background(), uid, orgID)
 | |
| 					assert.NoError(t, err)
 | |
| 				}
 | |
| 			})
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	t.Run("Should delete folders", func(t *testing.T) {
 | |
| 		featuresFlagOff := featuremgmt.WithFeatures()
 | |
| 		serviceWithFlagOff := &Service{
 | |
| 			cfg:                  cfg,
 | |
| 			log:                  log.New("test-folder-service"),
 | |
| 			dashboardFolderStore: folderStore,
 | |
| 			features:             featuresFlagOff,
 | |
| 			bus:                  b,
 | |
| 			db:                   db,
 | |
| 			registry:             make(map[string]folder.RegistryService),
 | |
| 		}
 | |
| 
 | |
| 		testCases := []struct {
 | |
| 			service           *Service
 | |
| 			featuresFlag      *featuremgmt.FeatureManager
 | |
| 			prefix            string
 | |
| 			depth             int
 | |
| 			forceDelete       bool
 | |
| 			deletionErr       error
 | |
| 			dashboardErr      error
 | |
| 			folderErr         error
 | |
| 			libPanelParentErr error
 | |
| 			libPanelSubErr    error
 | |
| 			desc              string
 | |
| 		}{
 | |
| 			{
 | |
| 				service:           serviceWithFlagOn,
 | |
| 				featuresFlag:      featuresFlagOn,
 | |
| 				prefix:            "flagon-force",
 | |
| 				depth:             3,
 | |
| 				forceDelete:       true,
 | |
| 				dashboardErr:      dashboards.ErrFolderNotFound,
 | |
| 				folderErr:         folder.ErrFolderNotFound,
 | |
| 				libPanelParentErr: model.ErrLibraryElementNotFound,
 | |
| 				libPanelSubErr:    model.ErrLibraryElementNotFound,
 | |
| 				desc:              "With nested folder feature flag on and force deletion of rules",
 | |
| 			},
 | |
| 			{
 | |
| 				service:      serviceWithFlagOn,
 | |
| 				featuresFlag: featuresFlagOn,
 | |
| 				prefix:       "flagon-noforce",
 | |
| 				depth:        3,
 | |
| 				forceDelete:  false,
 | |
| 				deletionErr:  dashboards.ErrFolderContainsAlertRules,
 | |
| 				desc:         "With nested folder feature flag on and no force deletion of rules",
 | |
| 			},
 | |
| 			{
 | |
| 				service:           serviceWithFlagOff,
 | |
| 				featuresFlag:      featuresFlagOff,
 | |
| 				prefix:            "flagoff-force",
 | |
| 				depth:             1,
 | |
| 				forceDelete:       true,
 | |
| 				dashboardErr:      dashboards.ErrFolderNotFound,
 | |
| 				libPanelParentErr: model.ErrLibraryElementNotFound,
 | |
| 				desc:              "With nested folder feature flag off and force deletion of rules",
 | |
| 			},
 | |
| 			{
 | |
| 				service:      serviceWithFlagOff,
 | |
| 				featuresFlag: featuresFlagOff,
 | |
| 				prefix:       "flagoff-noforce",
 | |
| 				depth:        1,
 | |
| 				forceDelete:  false,
 | |
| 				deletionErr:  dashboards.ErrFolderContainsAlertRules,
 | |
| 				desc:         "With nested folder feature flag off and no force deletion of rules",
 | |
| 			},
 | |
| 		}
 | |
| 
 | |
| 		for _, tc := range testCases {
 | |
| 			t.Run(tc.desc, func(t *testing.T) {
 | |
| 				origNewGuardian := guardian.New
 | |
| 				guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{
 | |
| 					CanSaveValue: true,
 | |
| 					CanViewValue: true,
 | |
| 					// CanEditValue is required to create library elements
 | |
| 					CanEditValue: true,
 | |
| 				})
 | |
| 
 | |
| 				elementService := libraryelements.ProvideService(cfg, db, routeRegister, tc.service, tc.featuresFlag)
 | |
| 				lps, err := librarypanels.ProvideService(cfg, db, routeRegister, elementService, tc.service)
 | |
| 				require.NoError(t, err)
 | |
| 
 | |
| 				dashStore, err := database.ProvideDashboardStore(db, db.Cfg, tc.featuresFlag, tagimpl.ProvideService(db, db.Cfg), quotaService)
 | |
| 				require.NoError(t, err)
 | |
| 				nestedFolderStore := ProvideStore(db, db.Cfg, tc.featuresFlag)
 | |
| 				tc.service.dashboardStore = dashStore
 | |
| 				tc.service.store = nestedFolderStore
 | |
| 
 | |
| 				dashSrv, err := service.ProvideDashboardServiceImpl(cfg, dashStore, folderStore, nil, tc.featuresFlag, folderPermissions, dashboardPermissions, ac, tc.service)
 | |
| 				require.NoError(t, err)
 | |
| 				alertStore, err := ngstore.ProvideDBStore(cfg, tc.featuresFlag, db, tc.service, ac, dashSrv)
 | |
| 				require.NoError(t, err)
 | |
| 
 | |
| 				ancestorUIDs := CreateSubtreeInStore(t, nestedFolderStore, serviceWithFlagOn, tc.depth, tc.prefix, createCmd)
 | |
| 
 | |
| 				parent, err := serviceWithFlagOn.dashboardFolderStore.GetFolderByUID(context.Background(), orgID, ancestorUIDs[0])
 | |
| 				require.NoError(t, err)
 | |
| 				_ = createRule(t, alertStore, parent.UID, "parent alert")
 | |
| 
 | |
| 				var (
 | |
| 					subfolder *folder.Folder
 | |
| 					subPanel  model.LibraryElementDTO
 | |
| 				)
 | |
| 				if tc.depth > 1 {
 | |
| 					subfolder, err = serviceWithFlagOn.dashboardFolderStore.GetFolderByUID(context.Background(), orgID, ancestorUIDs[1])
 | |
| 					require.NoError(t, err)
 | |
| 					_ = createRule(t, alertStore, subfolder.UID, "sub alert")
 | |
| 					libraryElementCmd.FolderID = subfolder.ID
 | |
| 					subPanel, err = lps.LibraryElementService.CreateElement(context.Background(), &signedInUser, libraryElementCmd)
 | |
| 					require.NoError(t, err)
 | |
| 				}
 | |
| 
 | |
| 				libraryElementCmd.FolderID = parent.ID
 | |
| 				parentPanel, err := lps.LibraryElementService.CreateElement(context.Background(), &signedInUser, libraryElementCmd)
 | |
| 				require.NoError(t, err)
 | |
| 
 | |
| 				deleteCmd := folder.DeleteFolderCommand{
 | |
| 					UID:              ancestorUIDs[0],
 | |
| 					OrgID:            orgID,
 | |
| 					SignedInUser:     &signedInUser,
 | |
| 					ForceDeleteRules: tc.forceDelete,
 | |
| 				}
 | |
| 
 | |
| 				err = tc.service.Delete(context.Background(), &deleteCmd)
 | |
| 				require.ErrorIs(t, err, tc.deletionErr)
 | |
| 
 | |
| 				for i, uid := range ancestorUIDs {
 | |
| 					// dashboard table
 | |
| 					_, err := tc.service.dashboardFolderStore.GetFolderByUID(context.Background(), orgID, uid)
 | |
| 					require.ErrorIs(t, err, tc.dashboardErr)
 | |
| 					// folder table
 | |
| 					_, err = tc.service.store.Get(context.Background(), folder.GetFolderQuery{UID: &ancestorUIDs[i], OrgID: orgID})
 | |
| 					require.ErrorIs(t, err, tc.folderErr)
 | |
| 				}
 | |
| 
 | |
| 				_, err = lps.LibraryElementService.GetElement(context.Background(), &signedInUser, model.GetLibraryElementCommand{
 | |
| 					FolderName: parent.Title,
 | |
| 					FolderID:   parent.ID,
 | |
| 					UID:        parentPanel.UID,
 | |
| 				})
 | |
| 				require.ErrorIs(t, err, tc.libPanelParentErr)
 | |
| 				if tc.depth > 1 {
 | |
| 					_, err = lps.LibraryElementService.GetElement(context.Background(), &signedInUser, model.GetLibraryElementCommand{
 | |
| 						FolderName: subfolder.Title,
 | |
| 						FolderID:   subfolder.ID,
 | |
| 						UID:        subPanel.UID,
 | |
| 					})
 | |
| 					require.ErrorIs(t, err, tc.libPanelSubErr)
 | |
| 				}
 | |
| 				t.Cleanup(func() {
 | |
| 					guardian.New = origNewGuardian
 | |
| 				})
 | |
| 			})
 | |
| 		}
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestNestedFolderServiceFeatureToggle(t *testing.T) {
 | |
| 	g := guardian.New
 | |
| 	guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 	t.Cleanup(func() {
 | |
| 		guardian.New = g
 | |
| 	})
 | |
| 
 | |
| 	nestedFolderStore := NewFakeStore()
 | |
| 
 | |
| 	dashStore := dashboards.FakeDashboardStore{}
 | |
| 	dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 	dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(&dashboards.Dashboard{}, nil)
 | |
| 
 | |
| 	dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 	dashboardFolderStore.On("GetFolderByID", mock.Anything, mock.AnythingOfType("int64"), mock.AnythingOfType("int64")).Return(&folder.Folder{}, nil)
 | |
| 
 | |
| 	cfg := setting.NewCfg()
 | |
| 	folderService := &Service{
 | |
| 		cfg:                  cfg,
 | |
| 		store:                nestedFolderStore,
 | |
| 		dashboardStore:       &dashStore,
 | |
| 		dashboardFolderStore: dashboardFolderStore,
 | |
| 		features:             featuremgmt.WithFeatures(featuremgmt.FlagNestedFolders),
 | |
| 		log:                  log.New("test-folder-service"),
 | |
| 		accessControl:        acimpl.ProvideAccessControl(cfg),
 | |
| 	}
 | |
| 	t.Run("create folder", func(t *testing.T) {
 | |
| 		nestedFolderStore.ExpectedFolder = &folder.Folder{ParentUID: util.GenerateShortUID()}
 | |
| 		res, err := folderService.Create(context.Background(), &folder.CreateFolderCommand{SignedInUser: usr, Title: "my folder"})
 | |
| 		require.NoError(t, err)
 | |
| 		require.NotNil(t, res.UID)
 | |
| 		require.NotEmpty(t, res.ParentUID)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestNestedFolderService(t *testing.T) {
 | |
| 	t.Run("with feature flag unset", func(t *testing.T) {
 | |
| 		t.Run("When create folder, no create in folder table done", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			// dashboard store & service commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 			dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(&dashboards.Dashboard{}, nil)
 | |
| 
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 			dashboardFolderStore.On("GetFolderByID", mock.Anything, mock.AnythingOfType("int64"), mock.AnythingOfType("int64")).Return(&folder.Folder{}, nil)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures(), acimpl.ProvideAccessControl(setting.NewCfg()), dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 				OrgID:        orgID,
 | |
| 				Title:        "myFolder",
 | |
| 				UID:          "myFolder",
 | |
| 				SignedInUser: usr,
 | |
| 			})
 | |
| 			require.NoError(t, err)
 | |
| 			// CreateFolder should not call the folder store create if the feature toggle is not enabled.
 | |
| 			require.False(t, nestedFolderStore.CreateCalled)
 | |
| 		})
 | |
| 	})
 | |
| 
 | |
| 	t.Run("with nested folder feature flag on", func(t *testing.T) {
 | |
| 		t.Run("Should be able to create a nested folder under the root", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			// dashboard store commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 			dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(&dashboards.Dashboard{}, nil)
 | |
| 
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 			dashboardFolderStore.On("GetFolderByID", mock.Anything, mock.AnythingOfType("int64"), mock.AnythingOfType("int64")).Return(&folder.Folder{}, nil)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), acimpl.ProvideAccessControl(setting.NewCfg()), dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 				OrgID:        orgID,
 | |
| 				Title:        "myFolder",
 | |
| 				UID:          "myFolder",
 | |
| 				SignedInUser: usr,
 | |
| 			})
 | |
| 			require.NoError(t, err)
 | |
| 			// CreateFolder should also call the folder store's create method.
 | |
| 			require.True(t, nestedFolderStore.CreateCalled)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("Should not be able to create new folder under another folder without the right permissions", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			dash := dashboards.NewDashboardFolder("Test-Folder")
 | |
| 			dash.ID = rand.Int63()
 | |
| 			dash.UID = "some_uid"
 | |
| 
 | |
| 			tempUser := &user.SignedInUser{UserID: 1, OrgID: orgID, Permissions: map[int64]map[string][]string{}}
 | |
| 			tempUser.Permissions[orgID] = map[string][]string{dashboards.ActionFoldersWrite: {dashboards.ScopeFoldersProvider.GetResourceScopeUID("wrong_uid")}}
 | |
| 
 | |
| 			// dashboard store commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 			dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(&dashboards.Dashboard{}, nil)
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, nil, nil, featuremgmt.WithFeatures("nestedFolders"), acimpl.ProvideAccessControl(setting.NewCfg()), dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 				OrgID:        orgID,
 | |
| 				Title:        dash.Title,
 | |
| 				UID:          dash.UID,
 | |
| 				SignedInUser: tempUser,
 | |
| 				ParentUID:    "some_parent",
 | |
| 			})
 | |
| 			require.ErrorIs(t, err, dashboards.ErrFolderAccessDenied)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("Should be able to create new folder under another folder with the right permissions", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			dash := dashboards.NewDashboardFolder("Test-Folder")
 | |
| 			dash.ID = rand.Int63()
 | |
| 			dash.UID = "some_uid"
 | |
| 
 | |
| 			// dashboard store commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 			dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(&dashboards.Dashboard{}, nil)
 | |
| 
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 			dashboardFolderStore.On("GetFolderByID", mock.Anything, mock.AnythingOfType("int64"), mock.AnythingOfType("int64")).Return(&folder.Folder{}, nil)
 | |
| 
 | |
| 			nestedFolderUser := &user.SignedInUser{UserID: 1, OrgID: orgID, Permissions: map[int64]map[string][]string{}}
 | |
| 			nestedFolderUser.Permissions[orgID] = map[string][]string{dashboards.ActionFoldersWrite: {dashboards.ScopeFoldersProvider.GetResourceScopeUID("some_parent")}}
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), acimpl.ProvideAccessControl(setting.NewCfg()), dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 				OrgID:        orgID,
 | |
| 				Title:        dash.Title,
 | |
| 				UID:          dash.UID,
 | |
| 				SignedInUser: nestedFolderUser,
 | |
| 				ParentUID:    "some_parent",
 | |
| 			})
 | |
| 			require.NoError(t, err)
 | |
| 			require.True(t, nestedFolderStore.CreateCalled)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("create without UID, no error", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			// dashboard store commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 			dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(&dashboards.Dashboard{UID: "newUID"}, nil)
 | |
| 
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 			dashboardFolderStore.On("GetFolderByID", mock.Anything, mock.AnythingOfType("int64"), mock.AnythingOfType("int64")).Return(&folder.Folder{}, nil)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), actest.FakeAccessControl{
 | |
| 				ExpectedEvaluate: true,
 | |
| 			}, dbtest.NewFakeDB())
 | |
| 			f, err := folderSvc.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 				OrgID:        orgID,
 | |
| 				Title:        "myFolder",
 | |
| 				SignedInUser: usr,
 | |
| 			})
 | |
| 			require.NoError(t, err)
 | |
| 			// CreateFolder should also call the folder store's create method.
 | |
| 			require.True(t, nestedFolderStore.CreateCalled)
 | |
| 			require.Equal(t, "newUID", f.UID)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("create failed because of circular reference", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			dashboardFolder := dashboards.NewDashboardFolder("myFolder")
 | |
| 			dashboardFolder.ID = rand.Int63()
 | |
| 			dashboardFolder.UID = "myFolder"
 | |
| 			f := dashboards.FromDashboard(dashboardFolder)
 | |
| 
 | |
| 			// dashboard store commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 			dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(dashboardFolder, nil)
 | |
| 			var actualCmd *dashboards.DeleteDashboardCommand
 | |
| 			dashStore.On("DeleteDashboard", mock.Anything, mock.Anything).Run(func(args mock.Arguments) {
 | |
| 				actualCmd = args.Get(1).(*dashboards.DeleteDashboardCommand)
 | |
| 			}).Return(nil).Once()
 | |
| 
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 			dashboardFolderStore.On("GetFolderByID", mock.Anything, orgID, dashboardFolder.ID).Return(f, nil)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedParentFolders = []*folder.Folder{
 | |
| 				{UID: "newFolder", ParentUID: "newFolder"},
 | |
| 				{UID: "newFolder2", ParentUID: "newFolder2"},
 | |
| 				{UID: "newFolder3", ParentUID: "newFolder3"},
 | |
| 				{UID: "myFolder", ParentUID: "newFolder"},
 | |
| 			}
 | |
| 
 | |
| 			cmd := folder.CreateFolderCommand{
 | |
| 				ParentUID:    "myFolder1",
 | |
| 				OrgID:        orgID,
 | |
| 				Title:        "myFolder",
 | |
| 				UID:          "myFolder",
 | |
| 				SignedInUser: usr,
 | |
| 			}
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), actest.FakeAccessControl{
 | |
| 				ExpectedEvaluate: true,
 | |
| 			}, dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Create(context.Background(), &cmd)
 | |
| 			require.Error(t, err, folder.ErrCircularReference)
 | |
| 			// CreateFolder should not call the folder store's create method.
 | |
| 			require.False(t, nestedFolderStore.CreateCalled)
 | |
| 			require.NotNil(t, actualCmd)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("create returns error from nested folder service", func(t *testing.T) {
 | |
| 			// This test creates and deletes the dashboard, so needs some extra setup.
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			// dashboard store commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil)
 | |
| 			dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(&dashboards.Dashboard{}, nil)
 | |
| 			var actualCmd *dashboards.DeleteDashboardCommand
 | |
| 			dashStore.On("DeleteDashboard", mock.Anything, mock.Anything).Run(func(args mock.Arguments) {
 | |
| 				actualCmd = args.Get(1).(*dashboards.DeleteDashboardCommand)
 | |
| 			}).Return(nil).Once()
 | |
| 
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 			dashboardFolderStore.On("GetFolderByID", mock.Anything, mock.AnythingOfType("int64"), mock.AnythingOfType("int64")).Return(&folder.Folder{}, nil)
 | |
| 
 | |
| 			// return an error from the folder store
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedError = errors.New("FAILED")
 | |
| 
 | |
| 			// the service return success as long as the legacy create succeeds
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), actest.FakeAccessControl{
 | |
| 				ExpectedEvaluate: true,
 | |
| 			}, dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 				OrgID:        orgID,
 | |
| 				Title:        "myFolder",
 | |
| 				UID:          "myFolder",
 | |
| 				SignedInUser: usr,
 | |
| 			})
 | |
| 			require.Error(t, err, "FAILED")
 | |
| 
 | |
| 			// CreateFolder should also call the folder store's create method.
 | |
| 			require.True(t, nestedFolderStore.CreateCalled)
 | |
| 			require.NotNil(t, actualCmd)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("move without the right permissions should fail", func(t *testing.T) {
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedFolder = &folder.Folder{UID: "myFolder", ParentUID: "newFolder"}
 | |
| 
 | |
| 			nestedFolderUser := &user.SignedInUser{UserID: 1, OrgID: orgID, Permissions: map[int64]map[string][]string{}}
 | |
| 			nestedFolderUser.Permissions[orgID] = map[string][]string{dashboards.ActionFoldersWrite: {dashboards.ScopeFoldersProvider.GetResourceScopeUID("wrong_uid")}}
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), acimpl.ProvideAccessControl(setting.NewCfg()), dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Move(context.Background(), &folder.MoveFolderCommand{UID: "myFolder", NewParentUID: "newFolder", OrgID: orgID, SignedInUser: nestedFolderUser})
 | |
| 			require.ErrorIs(t, err, dashboards.ErrFolderAccessDenied)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("move with the right permissions succeeds", func(t *testing.T) {
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedFolder = &folder.Folder{UID: "myFolder", ParentUID: "newFolder"}
 | |
| 			nestedFolderStore.ExpectedParentFolders = []*folder.Folder{
 | |
| 				{UID: "newFolder", ParentUID: "newFolder"},
 | |
| 				{UID: "newFolder2", ParentUID: "newFolder2"},
 | |
| 				{UID: "newFolder3", ParentUID: "newFolder3"},
 | |
| 			}
 | |
| 
 | |
| 			nestedFolderUser := &user.SignedInUser{UserID: 1, OrgID: orgID, Permissions: map[int64]map[string][]string{}}
 | |
| 			nestedFolderUser.Permissions[orgID] = map[string][]string{dashboards.ActionFoldersWrite: {dashboards.ScopeFoldersProvider.GetResourceScopeUID("newFolder")}}
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), acimpl.ProvideAccessControl(setting.NewCfg()), dbtest.NewFakeDB())
 | |
| 			f, err := folderSvc.Move(context.Background(), &folder.MoveFolderCommand{UID: "myFolder", NewParentUID: "newFolder", OrgID: orgID, SignedInUser: nestedFolderUser})
 | |
| 			require.NoError(t, err)
 | |
| 			require.NotNil(t, f)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("move to the root folder without folder creation permissions fails", func(t *testing.T) {
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedFolder = &folder.Folder{UID: "myFolder", ParentUID: "newFolder"}
 | |
| 
 | |
| 			nestedFolderUser := &user.SignedInUser{UserID: 1, OrgID: orgID, Permissions: map[int64]map[string][]string{}}
 | |
| 			nestedFolderUser.Permissions[orgID] = map[string][]string{dashboards.ActionFoldersWrite: {dashboards.ScopeFoldersProvider.GetResourceScopeUID("")}}
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), acimpl.ProvideAccessControl(setting.NewCfg()), dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Move(context.Background(), &folder.MoveFolderCommand{UID: "myFolder", NewParentUID: "", OrgID: orgID, SignedInUser: nestedFolderUser})
 | |
| 			require.Error(t, err, dashboards.ErrFolderAccessDenied)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("move to the root folder with folder creation permissions succeeds", func(t *testing.T) {
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedFolder = &folder.Folder{UID: "myFolder", ParentUID: "newFolder"}
 | |
| 			nestedFolderStore.ExpectedParentFolders = []*folder.Folder{
 | |
| 				{UID: "newFolder", ParentUID: "newFolder"},
 | |
| 				{UID: "newFolder2", ParentUID: "newFolder2"},
 | |
| 				{UID: "newFolder3", ParentUID: "newFolder3"},
 | |
| 			}
 | |
| 
 | |
| 			nestedFolderUser := &user.SignedInUser{UserID: 1, OrgID: orgID, Permissions: map[int64]map[string][]string{}}
 | |
| 			nestedFolderUser.Permissions[orgID] = map[string][]string{dashboards.ActionFoldersCreate: {}}
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), acimpl.ProvideAccessControl(setting.NewCfg()), dbtest.NewFakeDB())
 | |
| 			f, err := folderSvc.Move(context.Background(), &folder.MoveFolderCommand{UID: "myFolder", NewParentUID: "", OrgID: orgID, SignedInUser: nestedFolderUser})
 | |
| 			require.NoError(t, err)
 | |
| 			require.NotNil(t, f)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("move when parentUID in the current subtree returns error from nested folder service", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true, CanViewValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedFolder = &folder.Folder{UID: "myFolder", ParentUID: "newFolder"}
 | |
| 			nestedFolderStore.ExpectedError = folder.ErrCircularReference
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), actest.FakeAccessControl{
 | |
| 				ExpectedEvaluate: true,
 | |
| 			}, dbtest.NewFakeDB())
 | |
| 			f, err := folderSvc.Move(context.Background(), &folder.MoveFolderCommand{UID: "myFolder", NewParentUID: "newFolder", OrgID: orgID, SignedInUser: usr})
 | |
| 			require.Error(t, err, folder.ErrCircularReference)
 | |
| 			require.Nil(t, f)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("move when new parentUID depth + subTree height bypassed maximum depth returns error", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true, CanViewValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedFolder = &folder.Folder{UID: "myFolder", ParentUID: "newFolder"}
 | |
| 			nestedFolderStore.ExpectedParentFolders = []*folder.Folder{
 | |
| 				{UID: "newFolder", ParentUID: "newFolder"},
 | |
| 				{UID: "newFolder2", ParentUID: "newFolder2"},
 | |
| 			}
 | |
| 			nestedFolderStore.ExpectedFolderHeight = 5
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), actest.FakeAccessControl{
 | |
| 				ExpectedEvaluate: true,
 | |
| 			}, dbtest.NewFakeDB())
 | |
| 			f, err := folderSvc.Move(context.Background(), &folder.MoveFolderCommand{UID: "myFolder", NewParentUID: "newFolder2", OrgID: orgID, SignedInUser: usr})
 | |
| 			require.Error(t, err, folder.ErrMaximumDepthReached)
 | |
| 			require.Nil(t, f)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("move when parentUID in the current subtree returns error from nested folder service", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true, CanViewValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedFolder = &folder.Folder{UID: "myFolder", ParentUID: "newFolder"}
 | |
| 			nestedFolderStore.ExpectedParentFolders = []*folder.Folder{{UID: "myFolder", ParentUID: "12345"}, {UID: "12345", ParentUID: ""}}
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), actest.FakeAccessControl{
 | |
| 				ExpectedEvaluate: true,
 | |
| 			}, dbtest.NewFakeDB())
 | |
| 			f, err := folderSvc.Move(context.Background(), &folder.MoveFolderCommand{UID: "myFolder", NewParentUID: "newFolder2", OrgID: orgID, SignedInUser: usr})
 | |
| 			require.Error(t, err, folder.ErrCircularReference)
 | |
| 			require.Nil(t, f)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("create returns error if maximum depth reached", func(t *testing.T) {
 | |
| 			// This test creates and deletes the dashboard, so needs some extra setup.
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			// dashboard store commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 			dashStore.On("ValidateDashboardBeforeSave", mock.Anything, mock.AnythingOfType("*dashboards.Dashboard"), mock.AnythingOfType("bool")).Return(true, nil).Times(2)
 | |
| 			dashStore.On("SaveDashboard", mock.Anything, mock.AnythingOfType("dashboards.SaveDashboardCommand")).Return(&dashboards.Dashboard{}, nil)
 | |
| 			var actualCmd *dashboards.DeleteDashboardCommand
 | |
| 			dashStore.On("DeleteDashboard", mock.Anything, mock.Anything).Run(func(args mock.Arguments) {
 | |
| 				actualCmd = args.Get(1).(*dashboards.DeleteDashboardCommand)
 | |
| 			}).Return(nil).Once()
 | |
| 
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 			dashboardFolderStore.On("GetFolderByID", mock.Anything, mock.AnythingOfType("int64"), mock.AnythingOfType("int64")).Return(&folder.Folder{}, nil)
 | |
| 
 | |
| 			parents := make([]*folder.Folder, 0, folder.MaxNestedFolderDepth)
 | |
| 			for i := 0; i < folder.MaxNestedFolderDepth; i++ {
 | |
| 				parents = append(parents, &folder.Folder{UID: fmt.Sprintf("folder%d", i)})
 | |
| 			}
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			//nestedFolderStore.ExpectedFolder = &folder.Folder{UID: "myFolder", ParentUID: "newFolder"}
 | |
| 			nestedFolderStore.ExpectedParentFolders = parents
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), actest.FakeAccessControl{
 | |
| 				ExpectedEvaluate: true,
 | |
| 			}, dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Create(context.Background(), &folder.CreateFolderCommand{
 | |
| 				Title:        "folder",
 | |
| 				OrgID:        orgID,
 | |
| 				ParentUID:    parents[len(parents)-1].UID,
 | |
| 				UID:          util.GenerateShortUID(),
 | |
| 				SignedInUser: usr,
 | |
| 			})
 | |
| 			assert.ErrorIs(t, err, folder.ErrMaximumDepthReached)
 | |
| 			require.NotNil(t, actualCmd)
 | |
| 		})
 | |
| 
 | |
| 		t.Run("get default folder, no error", func(t *testing.T) {
 | |
| 			g := guardian.New
 | |
| 			guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanSaveValue: true})
 | |
| 			t.Cleanup(func() {
 | |
| 				guardian.New = g
 | |
| 			})
 | |
| 
 | |
| 			// dashboard store commands that should be called.
 | |
| 			dashStore := &dashboards.FakeDashboardStore{}
 | |
| 
 | |
| 			dashboardFolderStore := foldertest.NewFakeFolderStore(t)
 | |
| 
 | |
| 			nestedFolderStore := NewFakeStore()
 | |
| 			nestedFolderStore.ExpectedError = folder.ErrFolderNotFound
 | |
| 
 | |
| 			folderSvc := setup(t, dashStore, dashboardFolderStore, nestedFolderStore, featuremgmt.WithFeatures("nestedFolders"), actest.FakeAccessControl{
 | |
| 				ExpectedEvaluate: true,
 | |
| 			}, dbtest.NewFakeDB())
 | |
| 			_, err := folderSvc.Get(context.Background(), &folder.GetFolderQuery{
 | |
| 				OrgID:        orgID,
 | |
| 				ID:           &folder.GeneralFolder.ID,
 | |
| 				SignedInUser: usr,
 | |
| 			})
 | |
| 			require.NoError(t, err)
 | |
| 		})
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func CreateSubtreeInStore(t *testing.T, store *sqlStore, service *Service, depth int, prefix string, cmd folder.CreateFolderCommand) []string {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	ancestorUIDs := []string{}
 | |
| 	if cmd.ParentUID != "" {
 | |
| 		ancestorUIDs = append(ancestorUIDs, cmd.ParentUID)
 | |
| 	}
 | |
| 	for i := 0; i < depth; i++ {
 | |
| 		title := fmt.Sprintf("%sfolder-%d", prefix, i)
 | |
| 		cmd.Title = title
 | |
| 		cmd.UID = util.GenerateShortUID()
 | |
| 
 | |
| 		f, err := service.Create(context.Background(), &cmd)
 | |
| 		require.NoError(t, err)
 | |
| 		require.Equal(t, title, f.Title)
 | |
| 		require.NotEmpty(t, f.ID)
 | |
| 		require.NotEmpty(t, f.UID)
 | |
| 
 | |
| 		parents, err := store.GetParents(context.Background(), folder.GetParentsQuery{
 | |
| 			UID:   f.UID,
 | |
| 			OrgID: cmd.OrgID,
 | |
| 		})
 | |
| 		require.NoError(t, err)
 | |
| 		parentUIDs := []string{}
 | |
| 		for _, p := range parents {
 | |
| 			parentUIDs = append(parentUIDs, p.UID)
 | |
| 		}
 | |
| 		require.Equal(t, ancestorUIDs, parentUIDs)
 | |
| 
 | |
| 		ancestorUIDs = append(ancestorUIDs, f.UID)
 | |
| 
 | |
| 		cmd.ParentUID = f.UID
 | |
| 	}
 | |
| 
 | |
| 	return ancestorUIDs
 | |
| }
 | |
| 
 | |
| func setup(t *testing.T, dashStore dashboards.Store, dashboardFolderStore folder.FolderStore, nestedFolderStore store, features featuremgmt.FeatureToggles, ac accesscontrol.AccessControl, db db.DB) folder.Service {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	// nothing enabled yet
 | |
| 	cfg := setting.NewCfg()
 | |
| 	return &Service{
 | |
| 		cfg:                  cfg,
 | |
| 		log:                  log.New("test-folder-service"),
 | |
| 		dashboardStore:       dashStore,
 | |
| 		dashboardFolderStore: dashboardFolderStore,
 | |
| 		store:                nestedFolderStore,
 | |
| 		features:             features,
 | |
| 		accessControl:        ac,
 | |
| 		db:                   db,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func createRule(t *testing.T, store *ngstore.DBstore, folderUID, title string) *models.AlertRule {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	rule := models.AlertRule{
 | |
| 		OrgID:        orgID,
 | |
| 		NamespaceUID: folderUID,
 | |
| 		Title:        title,
 | |
| 		Updated:      time.Now(),
 | |
| 		UID:          util.GenerateShortUID(),
 | |
| 	}
 | |
| 	err := store.SQLStore.WithDbSession(context.Background(), func(sess *db.Session) error {
 | |
| 		_, err := sess.Table(models.AlertRule{}).InsertOne(rule)
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		return nil
 | |
| 	})
 | |
| 	require.NoError(t, err)
 | |
| 
 | |
| 	return &rule
 | |
| }
 |