package plugins import ( "bytes" "encoding/json" "errors" "io" "os" "path" "path/filepath" "reflect" "strings" "text/template" "github.com/grafana/grafana/pkg/log" "github.com/grafana/grafana/pkg/setting" "github.com/grafana/grafana/pkg/util" ) var ( DataSources map[string]*DataSourcePlugin Panels map[string]*PanelPlugin ApiPlugins map[string]*ApiPlugin StaticRoutes []*PluginStaticRoute Apps map[string]*AppPlugin PluginTypes map[string]interface{} ) type PluginScanner struct { pluginPath string errors []error } func Init() error { DataSources = make(map[string]*DataSourcePlugin) ApiPlugins = make(map[string]*ApiPlugin) StaticRoutes = make([]*PluginStaticRoute, 0) Panels = make(map[string]*PanelPlugin) Apps = make(map[string]*AppPlugin) PluginTypes = map[string]interface{}{ "panel": PanelPlugin{}, "datasource": DataSourcePlugin{}, "api": ApiPlugin{}, "app": AppPlugin{}, } scan(path.Join(setting.StaticRootPath, "app/plugins")) checkPluginPaths() // checkDependencies() return nil } // func checkDependencies() { // for appType, app := range Apps { // for _, reqPanel := range app.PanelPlugins { // if _, ok := Panels[reqPanel]; !ok { // log.Fatal(4, "App %s requires Panel type %s, but it is not present.", appType, reqPanel) // } // } // for _, reqDataSource := range app.DatasourcePlugins { // if _, ok := DataSources[reqDataSource]; !ok { // log.Fatal(4, "App %s requires DataSource type %s, but it is not present.", appType, reqDataSource) // } // } // for _, reqApiPlugin := range app.ApiPlugins { // if _, ok := ApiPlugins[reqApiPlugin]; !ok { // log.Fatal(4, "App %s requires ApiPlugin type %s, but it is not present.", appType, reqApiPlugin) // } // } // } // } func checkPluginPaths() error { for _, section := range setting.Cfg.Sections() { if strings.HasPrefix(section.Name(), "plugin.") { path := section.Key("path").String() if path != "" { log.Info("Plugin: Scaning dir %s", path) scan(path) } } } return nil } func scan(pluginDir string) error { scanner := &PluginScanner{ pluginPath: pluginDir, } if err := util.Walk(pluginDir, true, true, scanner.walker); err != nil { if pluginDir != "data/plugins" { log.Warn("Could not scan dir \"%v\" error: %s", pluginDir, err) } 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 interpolatePluginJson(reader io.Reader, pluginCommon *PluginBase) (io.Reader, error) { buf := new(bytes.Buffer) buf.ReadFrom(reader) jsonStr := buf.String() // tmpl, err := template.New("json").Parse(jsonStr) if err != nil { return nil, err } data := map[string]interface{}{ "PluginPublicRoot": "public/plugins/" + pluginCommon.Id, } var resultBuffer bytes.Buffer if err := tmpl.ExecuteTemplate(&resultBuffer, "json", data); err != nil { return nil, err } return bytes.NewReader(resultBuffer.Bytes()), 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) pluginCommon := PluginBase{} if err := jsonParser.Decode(&pluginCommon); err != nil { return err } if pluginCommon.Id == "" || pluginCommon.Type == "" { return errors.New("Did not find type and id property in plugin.json") } reader.Seek(0, 0) if newReader, err := interpolatePluginJson(reader, &pluginCommon); err != nil { return err } else { jsonParser = json.NewDecoder(newReader) } var loader PluginLoader if pluginGoType, exists := PluginTypes[pluginCommon.Type]; !exists { return errors.New("Unkown plugin type " + pluginCommon.Type) } else { loader = reflect.New(reflect.TypeOf(pluginGoType)).Interface().(PluginLoader) } return loader.Load(jsonParser, currentDir) }