package plugins import ( "encoding/json" "errors" "os" "path" "path/filepath" "strings" "github.com/grafana/grafana/pkg/log" "github.com/grafana/grafana/pkg/setting" ) var ( DataSources map[string]DataSourcePlugin ExternalPlugins []ExternalPlugin StaticRoutes []*StaticRootConfig ) type PluginScanner struct { pluginPath string errors []error } func Init() error { DataSources = make(map[string]DataSourcePlugin) ExternalPlugins = make([]ExternalPlugin, 0) StaticRoutes = make([]*StaticRootConfig, 0) scan(path.Join(setting.StaticRootPath, "app/plugins")) checkExternalPluginPaths() return nil } func checkExternalPluginPaths() error { for _, section := range setting.Cfg.Sections() { if strings.HasPrefix(section.Name(), "plugin.") { path := section.Key("path").String() if path != "" { log.Info("Plugin: scaning specific dir %s", path) scan(path) } } } return nil } func scan(pluginDir string) error { scanner := &PluginScanner{ pluginPath: pluginDir, } if err := filepath.Walk(pluginDir, scanner.walker); err != nil { return err } if len(scanner.errors) > 0 { return errors.New("Some plugins failed to load") } return nil } func (scanner *PluginScanner) walker(currentPath string, f os.FileInfo, err error) error { if err != nil { return err } if f.IsDir() { return nil } if f.Name() == "plugin.json" { err := scanner.loadPluginJson(currentPath) if err != nil { log.Error(3, "Failed to load plugin json file: %v, err: %v", currentPath, err) scanner.errors = append(scanner.errors, err) } } return nil } func (scanner *PluginScanner) loadPluginJson(pluginJsonFilePath string) error { currentDir := filepath.Dir(pluginJsonFilePath) reader, err := os.Open(pluginJsonFilePath) if err != nil { return err } defer reader.Close() jsonParser := json.NewDecoder(reader) pluginJson := make(map[string]interface{}) if err := jsonParser.Decode(&pluginJson); err != nil { return err } pluginType, exists := pluginJson["pluginType"] if !exists { return errors.New("Did not find pluginType property in plugin.json") } if pluginType == "datasource" { p := DataSourcePlugin{} reader.Seek(0, 0) if err := jsonParser.Decode(&p); err != nil { return err } if p.Type == "" { return errors.New("Did not find type property in plugin.json") } DataSources[p.Type] = p if p.StaticRootConfig != nil { p.StaticRootConfig.Path = path.Join(currentDir, p.StaticRootConfig.Path) StaticRoutes = append(StaticRoutes, p.StaticRootConfig) } } if pluginType == "externalPlugin" { p := ExternalPlugin{} reader.Seek(0, 0) if err := jsonParser.Decode(&p); err != nil { return err } ExternalPlugins = append(ExternalPlugins, p) } return nil }