file_reader_test.go 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. package dashboards
  2. import (
  3. "os"
  4. "path/filepath"
  5. "runtime"
  6. "testing"
  7. "time"
  8. "github.com/grafana/grafana/pkg/bus"
  9. "github.com/grafana/grafana/pkg/models"
  10. "github.com/grafana/grafana/pkg/services/dashboards"
  11. "github.com/grafana/grafana/pkg/log"
  12. . "github.com/smartystreets/goconvey/convey"
  13. )
  14. var (
  15. defaultDashboards = "testdata/test-dashboards/folder-one"
  16. brokenDashboards = "testdata/test-dashboards/broken-dashboards"
  17. oneDashboard = "testdata/test-dashboards/one-dashboard"
  18. containingId = "testdata/test-dashboards/containing-id"
  19. fakeService *fakeDashboardProvisioningService
  20. )
  21. func TestCreatingNewDashboardFileReader(t *testing.T) {
  22. Convey("creating new dashboard file reader", t, func() {
  23. cfg := &DashboardsAsConfig{
  24. Name: "Default",
  25. Type: "file",
  26. OrgId: 1,
  27. Folder: "",
  28. Options: map[string]interface{}{},
  29. }
  30. Convey("using path parameter", func() {
  31. cfg.Options["path"] = defaultDashboards
  32. reader, err := NewDashboardFileReader(cfg, log.New("test-logger"))
  33. So(err, ShouldBeNil)
  34. So(reader.Path, ShouldNotEqual, "")
  35. })
  36. Convey("using folder as options", func() {
  37. cfg.Options["folder"] = defaultDashboards
  38. reader, err := NewDashboardFileReader(cfg, log.New("test-logger"))
  39. So(err, ShouldBeNil)
  40. So(reader.Path, ShouldNotEqual, "")
  41. })
  42. Convey("using full path", func() {
  43. fullPath := "/var/lib/grafana/dashboards"
  44. if runtime.GOOS == "windows" {
  45. fullPath = `c:\var\lib\grafana`
  46. }
  47. cfg.Options["folder"] = fullPath
  48. reader, err := NewDashboardFileReader(cfg, log.New("test-logger"))
  49. So(err, ShouldBeNil)
  50. So(reader.Path, ShouldEqual, fullPath)
  51. So(filepath.IsAbs(reader.Path), ShouldBeTrue)
  52. })
  53. Convey("using relative path", func() {
  54. cfg.Options["folder"] = defaultDashboards
  55. reader, err := NewDashboardFileReader(cfg, log.New("test-logger"))
  56. So(err, ShouldBeNil)
  57. resolvedPath := reader.resolvePath(reader.Path)
  58. So(filepath.IsAbs(resolvedPath), ShouldBeTrue)
  59. })
  60. })
  61. }
  62. func TestDashboardFileReader(t *testing.T) {
  63. Convey("Dashboard file reader", t, func() {
  64. bus.ClearBusHandlers()
  65. origNewDashboardProvisioningService := dashboards.NewProvisioningService
  66. fakeService = mockDashboardProvisioningService()
  67. bus.AddHandler("test", mockGetDashboardQuery)
  68. logger := log.New("test.logger")
  69. Convey("Reading dashboards from disk", func() {
  70. cfg := &DashboardsAsConfig{
  71. Name: "Default",
  72. Type: "file",
  73. OrgId: 1,
  74. Folder: "",
  75. Options: map[string]interface{}{},
  76. }
  77. Convey("Can read default dashboard", func() {
  78. cfg.Options["path"] = defaultDashboards
  79. cfg.Folder = "Team A"
  80. reader, err := NewDashboardFileReader(cfg, logger)
  81. So(err, ShouldBeNil)
  82. err = reader.startWalkingDisk()
  83. So(err, ShouldBeNil)
  84. folders := 0
  85. dashboards := 0
  86. for _, i := range fakeService.inserted {
  87. if i.Dashboard.IsFolder {
  88. folders++
  89. } else {
  90. dashboards++
  91. }
  92. }
  93. So(folders, ShouldEqual, 1)
  94. So(dashboards, ShouldEqual, 2)
  95. })
  96. Convey("Can read default dashboard and replace old version in database", func() {
  97. cfg.Options["path"] = oneDashboard
  98. stat, _ := os.Stat(oneDashboard + "/dashboard1.json")
  99. fakeService.getDashboard = append(fakeService.getDashboard, &models.Dashboard{
  100. Updated: stat.ModTime().AddDate(0, 0, -1),
  101. Slug: "grafana",
  102. })
  103. reader, err := NewDashboardFileReader(cfg, logger)
  104. So(err, ShouldBeNil)
  105. err = reader.startWalkingDisk()
  106. So(err, ShouldBeNil)
  107. So(len(fakeService.inserted), ShouldEqual, 1)
  108. })
  109. Convey("Overrides id from dashboard.json files", func() {
  110. cfg.Options["path"] = containingId
  111. reader, err := NewDashboardFileReader(cfg, logger)
  112. So(err, ShouldBeNil)
  113. err = reader.startWalkingDisk()
  114. So(err, ShouldBeNil)
  115. So(len(fakeService.inserted), ShouldEqual, 1)
  116. })
  117. Convey("Invalid configuration should return error", func() {
  118. cfg := &DashboardsAsConfig{
  119. Name: "Default",
  120. Type: "file",
  121. OrgId: 1,
  122. Folder: "",
  123. }
  124. _, err := NewDashboardFileReader(cfg, logger)
  125. So(err, ShouldNotBeNil)
  126. })
  127. Convey("Broken dashboards should not cause error", func() {
  128. cfg.Options["path"] = brokenDashboards
  129. _, err := NewDashboardFileReader(cfg, logger)
  130. So(err, ShouldBeNil)
  131. })
  132. })
  133. Convey("Should not create new folder if folder name is missing", func() {
  134. cfg := &DashboardsAsConfig{
  135. Name: "Default",
  136. Type: "file",
  137. OrgId: 1,
  138. Folder: "",
  139. Options: map[string]interface{}{
  140. "folder": defaultDashboards,
  141. },
  142. }
  143. _, err := getOrCreateFolderId(cfg, fakeService)
  144. So(err, ShouldEqual, ErrFolderNameMissing)
  145. })
  146. Convey("can get or Create dashboard folder", func() {
  147. cfg := &DashboardsAsConfig{
  148. Name: "Default",
  149. Type: "file",
  150. OrgId: 1,
  151. Folder: "TEAM A",
  152. Options: map[string]interface{}{
  153. "folder": defaultDashboards,
  154. },
  155. }
  156. folderId, err := getOrCreateFolderId(cfg, fakeService)
  157. So(err, ShouldBeNil)
  158. inserted := false
  159. for _, d := range fakeService.inserted {
  160. if d.Dashboard.IsFolder && d.Dashboard.Id == folderId {
  161. inserted = true
  162. }
  163. }
  164. So(len(fakeService.inserted), ShouldEqual, 1)
  165. So(inserted, ShouldBeTrue)
  166. })
  167. Convey("Walking the folder with dashboards", func() {
  168. noFiles := map[string]os.FileInfo{}
  169. Convey("should skip dirs that starts with .", func() {
  170. shouldSkip := createWalkFn(noFiles)("path", &FakeFileInfo{isDirectory: true, name: ".folder"}, nil)
  171. So(shouldSkip, ShouldEqual, filepath.SkipDir)
  172. })
  173. Convey("should keep walking if file is not .json", func() {
  174. shouldSkip := createWalkFn(noFiles)("path", &FakeFileInfo{isDirectory: true, name: "folder"}, nil)
  175. So(shouldSkip, ShouldBeNil)
  176. })
  177. })
  178. Reset(func() {
  179. dashboards.NewProvisioningService = origNewDashboardProvisioningService
  180. })
  181. })
  182. }
  183. type FakeFileInfo struct {
  184. isDirectory bool
  185. name string
  186. }
  187. func (ffi *FakeFileInfo) IsDir() bool {
  188. return ffi.isDirectory
  189. }
  190. func (ffi FakeFileInfo) Size() int64 {
  191. return 1
  192. }
  193. func (ffi FakeFileInfo) Mode() os.FileMode {
  194. return 0777
  195. }
  196. func (ffi FakeFileInfo) Name() string {
  197. return ffi.name
  198. }
  199. func (ffi FakeFileInfo) ModTime() time.Time {
  200. return time.Time{}
  201. }
  202. func (ffi FakeFileInfo) Sys() interface{} {
  203. return nil
  204. }
  205. func mockDashboardProvisioningService() *fakeDashboardProvisioningService {
  206. mock := fakeDashboardProvisioningService{}
  207. dashboards.NewProvisioningService = func() dashboards.DashboardProvisioningService {
  208. return &mock
  209. }
  210. return &mock
  211. }
  212. type fakeDashboardProvisioningService struct {
  213. inserted []*dashboards.SaveDashboardDTO
  214. provisioned []*models.DashboardProvisioning
  215. getDashboard []*models.Dashboard
  216. }
  217. func (s *fakeDashboardProvisioningService) GetProvisionedDashboardData(name string) ([]*models.DashboardProvisioning, error) {
  218. return s.provisioned, nil
  219. }
  220. func (s *fakeDashboardProvisioningService) SaveProvisionedDashboard(dto *dashboards.SaveDashboardDTO, provisioning *models.DashboardProvisioning) (*models.Dashboard, error) {
  221. s.inserted = append(s.inserted, dto)
  222. s.provisioned = append(s.provisioned, provisioning)
  223. return dto.Dashboard, nil
  224. }
  225. func (s *fakeDashboardProvisioningService) SaveFolderForProvisionedDashboards(dto *dashboards.SaveDashboardDTO) (*models.Dashboard, error) {
  226. s.inserted = append(s.inserted, dto)
  227. return dto.Dashboard, nil
  228. }
  229. func mockGetDashboardQuery(cmd *models.GetDashboardQuery) error {
  230. for _, d := range fakeService.getDashboard {
  231. if d.Slug == cmd.Slug {
  232. cmd.Result = d
  233. return nil
  234. }
  235. }
  236. return models.ErrDashboardNotFound
  237. }