| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | package plugins | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"encoding/json" | 
					
						
							|  |  |  | 	"errors" | 
					
						
							| 
									
										
										
										
											2016-03-14 02:21:44 +08:00
										 |  |  | 	"io/ioutil" | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 	"os" | 
					
						
							| 
									
										
										
										
											2015-02-28 05:29:00 +08:00
										 |  |  | 	"path" | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 	"path/filepath" | 
					
						
							| 
									
										
										
										
											2016-01-10 06:34:20 +08:00
										 |  |  | 	"reflect" | 
					
						
							| 
									
										
										
										
											2015-11-19 23:50:17 +08:00
										 |  |  | 	"strings" | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/grafana/grafana/pkg/log" | 
					
						
							| 
									
										
										
										
											2015-02-28 05:29:00 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/setting" | 
					
						
							| 
									
										
										
										
											2015-10-08 12:22:09 +08:00
										 |  |  | 	"github.com/grafana/grafana/pkg/util" | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ( | 
					
						
							| 
									
										
										
										
											2015-12-17 23:53:58 +08:00
										 |  |  | 	DataSources  map[string]*DataSourcePlugin | 
					
						
							|  |  |  | 	Panels       map[string]*PanelPlugin | 
					
						
							| 
									
										
										
										
											2016-01-09 15:12:27 +08:00
										 |  |  | 	StaticRoutes []*PluginStaticRoute | 
					
						
							| 
									
										
										
										
											2015-12-17 23:53:58 +08:00
										 |  |  | 	Apps         map[string]*AppPlugin | 
					
						
							| 
									
										
										
										
											2016-02-10 18:03:12 +08:00
										 |  |  | 	Plugins      map[string]*PluginBase | 
					
						
							| 
									
										
										
										
											2016-01-10 06:34:20 +08:00
										 |  |  | 	PluginTypes  map[string]interface{} | 
					
						
							| 
									
										
										
										
											2016-04-12 00:21:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	GrafanaLatestVersion string | 
					
						
							|  |  |  | 	GrafanaHasUpdate     bool | 
					
						
							| 
									
										
										
										
											2016-06-07 05:06:44 +08:00
										 |  |  | 	plog                 log.Logger | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type PluginScanner struct { | 
					
						
							|  |  |  | 	pluginPath string | 
					
						
							|  |  |  | 	errors     []error | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 23:50:17 +08:00
										 |  |  | func Init() error { | 
					
						
							| 
									
										
										
										
											2016-06-07 05:06:44 +08:00
										 |  |  | 	plog = log.New("plugins") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 23:53:58 +08:00
										 |  |  | 	DataSources = make(map[string]*DataSourcePlugin) | 
					
						
							| 
									
										
										
										
											2016-01-09 15:12:27 +08:00
										 |  |  | 	StaticRoutes = make([]*PluginStaticRoute, 0) | 
					
						
							| 
									
										
										
										
											2015-12-17 23:53:58 +08:00
										 |  |  | 	Panels = make(map[string]*PanelPlugin) | 
					
						
							|  |  |  | 	Apps = make(map[string]*AppPlugin) | 
					
						
							| 
									
										
										
										
											2016-02-10 18:03:12 +08:00
										 |  |  | 	Plugins = make(map[string]*PluginBase) | 
					
						
							| 
									
										
										
										
											2016-01-10 06:34:20 +08:00
										 |  |  | 	PluginTypes = map[string]interface{}{ | 
					
						
							|  |  |  | 		"panel":      PanelPlugin{}, | 
					
						
							|  |  |  | 		"datasource": DataSourcePlugin{}, | 
					
						
							|  |  |  | 		"app":        AppPlugin{}, | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2015-11-19 19:55:13 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-07 05:06:44 +08:00
										 |  |  | 	plog.Info("Starting plugin search") | 
					
						
							| 
									
										
										
										
											2015-02-28 05:29:00 +08:00
										 |  |  | 	scan(path.Join(setting.StaticRootPath, "app/plugins")) | 
					
						
							| 
									
										
										
										
											2016-02-10 18:03:12 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// check if plugins dir exists
 | 
					
						
							|  |  |  | 	if _, err := os.Stat(setting.PluginsPath); os.IsNotExist(err) { | 
					
						
							| 
									
										
										
										
											2016-06-07 05:06:44 +08:00
										 |  |  | 		plog.Warn("Plugin dir does not exist", "dir", setting.PluginsPath) | 
					
						
							| 
									
										
										
										
											2016-02-10 18:03:12 +08:00
										 |  |  | 		if err = os.MkdirAll(setting.PluginsPath, os.ModePerm); err != nil { | 
					
						
							| 
									
										
										
										
											2016-06-07 05:06:44 +08:00
										 |  |  | 			plog.Warn("Failed to create plugin dir", "dir", setting.PluginsPath, "error", err) | 
					
						
							| 
									
										
										
										
											2016-02-10 18:03:12 +08:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2016-06-07 05:06:44 +08:00
										 |  |  | 			plog.Info("Plugin dir created", "dir", setting.PluginsPath) | 
					
						
							| 
									
										
										
										
											2016-02-10 18:03:12 +08:00
										 |  |  | 			scan(setting.PluginsPath) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		scan(setting.PluginsPath) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// check plugin paths defined in config
 | 
					
						
							| 
									
										
										
										
											2015-12-17 23:53:58 +08:00
										 |  |  | 	checkPluginPaths() | 
					
						
							| 
									
										
										
										
											2016-02-10 05:10:36 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	for _, panel := range Panels { | 
					
						
							|  |  |  | 		panel.initFrontendPlugin() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for _, panel := range DataSources { | 
					
						
							|  |  |  | 		panel.initFrontendPlugin() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	for _, app := range Apps { | 
					
						
							|  |  |  | 		app.initApp() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-12 00:47:04 +08:00
										 |  |  | 	go StartPluginUpdateChecker() | 
					
						
							| 
									
										
										
										
											2016-07-08 00:11:03 +08:00
										 |  |  | 	go updateAppDashboards() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 23:50:17 +08:00
										 |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 23:53:58 +08:00
										 |  |  | func checkPluginPaths() error { | 
					
						
							| 
									
										
										
										
											2015-11-19 23:50:17 +08:00
										 |  |  | 	for _, section := range setting.Cfg.Sections() { | 
					
						
							|  |  |  | 		if strings.HasPrefix(section.Name(), "plugin.") { | 
					
						
							|  |  |  | 			path := section.Key("path").String() | 
					
						
							|  |  |  | 			if path != "" { | 
					
						
							|  |  |  | 				scan(path) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							| 
									
										
										
										
											2015-02-28 05:29:00 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func scan(pluginDir string) error { | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 	scanner := &PluginScanner{ | 
					
						
							|  |  |  | 		pluginPath: pluginDir, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-10-08 12:22:09 +08:00
										 |  |  | 	if err := util.Walk(pluginDir, true, true, scanner.walker); err != nil { | 
					
						
							| 
									
										
										
										
											2016-01-07 15:51:31 +08:00
										 |  |  | 		if pluginDir != "data/plugins" { | 
					
						
							|  |  |  | 			log.Warn("Could not scan dir \"%v\" error: %s", pluginDir, err) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if len(scanner.errors) > 0 { | 
					
						
							|  |  |  | 		return errors.New("Some plugins failed to load") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 23:50:17 +08:00
										 |  |  | func (scanner *PluginScanner) walker(currentPath string, f os.FileInfo, err error) error { | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-10 01:17:32 +08:00
										 |  |  | 	if f.Name() == "node_modules" { | 
					
						
							|  |  |  | 		return util.WalkSkipDir | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 	if f.IsDir() { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if f.Name() == "plugin.json" { | 
					
						
							| 
									
										
										
										
											2015-11-19 23:50:17 +08:00
										 |  |  | 		err := scanner.loadPluginJson(currentPath) | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2016-02-10 18:03:12 +08:00
										 |  |  | 			log.Error(3, "Plugins: Failed to load plugin json file: %v,  err: %v", currentPath, err) | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 			scanner.errors = append(scanner.errors, err) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-19 23:50:17 +08:00
										 |  |  | func (scanner *PluginScanner) loadPluginJson(pluginJsonFilePath string) error { | 
					
						
							|  |  |  | 	currentDir := filepath.Dir(pluginJsonFilePath) | 
					
						
							|  |  |  | 	reader, err := os.Open(pluginJsonFilePath) | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-28 05:29:00 +08:00
										 |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	defer reader.Close() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	jsonParser := json.NewDecoder(reader) | 
					
						
							| 
									
										
										
										
											2016-01-10 06:34:20 +08:00
										 |  |  | 	pluginCommon := PluginBase{} | 
					
						
							| 
									
										
										
										
											2016-01-09 06:15:44 +08:00
										 |  |  | 	if err := jsonParser.Decode(&pluginCommon); err != nil { | 
					
						
							| 
									
										
										
										
											2015-02-28 05:29:00 +08:00
										 |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-09 06:15:44 +08:00
										 |  |  | 	if pluginCommon.Id == "" || pluginCommon.Type == "" { | 
					
						
							|  |  |  | 		return errors.New("Did not find type and id property in plugin.json") | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 06:34:20 +08:00
										 |  |  | 	var loader PluginLoader | 
					
						
							|  |  |  | 	if pluginGoType, exists := PluginTypes[pluginCommon.Type]; !exists { | 
					
						
							| 
									
										
										
										
											2016-02-13 08:01:03 +08:00
										 |  |  | 		return errors.New("Unknown plugin type " + pluginCommon.Type) | 
					
						
							| 
									
										
										
										
											2016-01-10 06:34:20 +08:00
										 |  |  | 	} else { | 
					
						
							|  |  |  | 		loader = reflect.New(reflect.TypeOf(pluginGoType)).Interface().(PluginLoader) | 
					
						
							| 
									
										
										
										
											2015-12-03 15:52:37 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-20 01:18:53 +08:00
										 |  |  | 	reader.Seek(0, 0) | 
					
						
							| 
									
										
										
										
											2016-01-10 06:34:20 +08:00
										 |  |  | 	return loader.Load(jsonParser, currentDir) | 
					
						
							| 
									
										
										
										
											2015-02-27 20:45:00 +08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-03-14 02:21:44 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | func GetPluginReadme(pluginId string) ([]byte, error) { | 
					
						
							|  |  |  | 	plug, exists := Plugins[pluginId] | 
					
						
							|  |  |  | 	if !exists { | 
					
						
							|  |  |  | 		return nil, PluginNotFoundError{pluginId} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if plug.Readme != nil { | 
					
						
							|  |  |  | 		return plug.Readme, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	readmePath := filepath.Join(plug.PluginDir, "README.md") | 
					
						
							|  |  |  | 	if _, err := os.Stat(readmePath); os.IsNotExist(err) { | 
					
						
							|  |  |  | 		readmePath = filepath.Join(plug.PluginDir, "readme.md") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if _, err := os.Stat(readmePath); os.IsNotExist(err) { | 
					
						
							|  |  |  | 		plug.Readme = make([]byte, 0) | 
					
						
							|  |  |  | 		return plug.Readme, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if readmeBytes, err := ioutil.ReadFile(readmePath); err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		plug.Readme = readmeBytes | 
					
						
							|  |  |  | 		return plug.Readme, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |