plugins.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. package plugins
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "os"
  6. "path"
  7. "path/filepath"
  8. "strings"
  9. "github.com/grafana/grafana/pkg/log"
  10. "github.com/grafana/grafana/pkg/setting"
  11. "github.com/grafana/grafana/pkg/util"
  12. )
  13. var (
  14. DataSources map[string]DataSourcePlugin
  15. Panels map[string]PanelPlugin
  16. StaticRoutes []*StaticRootConfig
  17. )
  18. type PluginScanner struct {
  19. pluginPath string
  20. errors []error
  21. }
  22. func Init() error {
  23. DataSources = make(map[string]DataSourcePlugin)
  24. StaticRoutes = make([]*StaticRootConfig, 0)
  25. Panels = make(map[string]PanelPlugin)
  26. scan(path.Join(setting.StaticRootPath, "app/plugins"))
  27. scan(path.Join(setting.PluginsPath))
  28. checkExternalPluginPaths()
  29. return nil
  30. }
  31. func checkExternalPluginPaths() error {
  32. for _, section := range setting.Cfg.Sections() {
  33. if strings.HasPrefix(section.Name(), "plugin.") {
  34. path := section.Key("path").String()
  35. if path != "" {
  36. log.Info("Plugin: Scaning dir %s", path)
  37. scan(path)
  38. }
  39. }
  40. }
  41. return nil
  42. }
  43. func scan(pluginDir string) error {
  44. scanner := &PluginScanner{
  45. pluginPath: pluginDir,
  46. }
  47. if err := util.Walk(pluginDir, true, true, scanner.walker); err != nil {
  48. log.Warn("Failed to scan dir \"%v\" error: %s", pluginDir, err)
  49. return err
  50. }
  51. if len(scanner.errors) > 0 {
  52. return errors.New("Some plugins failed to load")
  53. }
  54. return nil
  55. }
  56. func (scanner *PluginScanner) walker(currentPath string, f os.FileInfo, err error) error {
  57. if err != nil {
  58. return err
  59. }
  60. if f.IsDir() {
  61. return nil
  62. }
  63. if f.Name() == "plugin.json" {
  64. err := scanner.loadPluginJson(currentPath)
  65. if err != nil {
  66. log.Error(3, "Failed to load plugin json file: %v, err: %v", currentPath, err)
  67. scanner.errors = append(scanner.errors, err)
  68. }
  69. }
  70. return nil
  71. }
  72. func addStaticRoot(staticRootConfig *StaticRootConfig, currentDir string) {
  73. if staticRootConfig != nil {
  74. staticRootConfig.Path = path.Join(currentDir, staticRootConfig.Path)
  75. StaticRoutes = append(StaticRoutes, staticRootConfig)
  76. }
  77. }
  78. func (scanner *PluginScanner) loadPluginJson(pluginJsonFilePath string) error {
  79. currentDir := filepath.Dir(pluginJsonFilePath)
  80. reader, err := os.Open(pluginJsonFilePath)
  81. if err != nil {
  82. return err
  83. }
  84. defer reader.Close()
  85. jsonParser := json.NewDecoder(reader)
  86. pluginJson := make(map[string]interface{})
  87. if err := jsonParser.Decode(&pluginJson); err != nil {
  88. return err
  89. }
  90. pluginType, exists := pluginJson["pluginType"]
  91. if !exists {
  92. return errors.New("Did not find pluginType property in plugin.json")
  93. }
  94. if pluginType == "datasource" {
  95. p := DataSourcePlugin{}
  96. reader.Seek(0, 0)
  97. if err := jsonParser.Decode(&p); err != nil {
  98. return err
  99. }
  100. if p.Type == "" {
  101. return errors.New("Did not find type property in plugin.json")
  102. }
  103. DataSources[p.Type] = p
  104. addStaticRoot(p.StaticRootConfig, currentDir)
  105. }
  106. if pluginType == "panel" {
  107. p := PanelPlugin{}
  108. reader.Seek(0, 0)
  109. if err := jsonParser.Decode(&p); err != nil {
  110. return err
  111. }
  112. if p.Type == "" {
  113. return errors.New("Did not find type property in plugin.json")
  114. }
  115. Panels[p.Type] = p
  116. addStaticRoot(p.StaticRootConfig, currentDir)
  117. }
  118. return nil
  119. }