install_command.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. package commands
  2. import (
  3. "archive/zip"
  4. "bytes"
  5. "errors"
  6. "github.com/grafana/grafana/pkg/cmd/grafana-cli/log"
  7. m "github.com/grafana/grafana/pkg/cmd/grafana-cli/models"
  8. services "github.com/grafana/grafana/pkg/cmd/grafana-cli/services"
  9. "io"
  10. "io/ioutil"
  11. "net/http"
  12. "os"
  13. "path"
  14. "regexp"
  15. )
  16. func validateInput(c CommandLine, pluginFolder string) error {
  17. arg := c.Args().First()
  18. if arg == "" {
  19. return errors.New("please specify plugin to install")
  20. }
  21. pluginDir := c.GlobalString("path")
  22. if pluginDir == "" {
  23. return errors.New("missing path flag")
  24. }
  25. fileinfo, err := os.Stat(pluginDir)
  26. if err != nil && !fileinfo.IsDir() {
  27. return errors.New("path is not a directory")
  28. }
  29. return nil
  30. }
  31. func installCommand(c CommandLine) error {
  32. pluginFolder := c.GlobalString("path")
  33. if err := validateInput(c, pluginFolder); err != nil {
  34. return err
  35. }
  36. pluginToInstall := c.Args().First()
  37. version := c.Args().Get(1)
  38. log.Infof("version: %v\n", version)
  39. return InstallPlugin(pluginToInstall, pluginFolder, version)
  40. }
  41. func InstallPlugin(pluginName, pluginFolder, version string) error {
  42. plugin, err := services.GetPlugin(pluginName)
  43. if err != nil {
  44. return err
  45. }
  46. v, err := SelectVersion(plugin, version)
  47. if err != nil {
  48. return err
  49. }
  50. url := v.Url
  51. commit := v.Commit
  52. downloadURL := url + "/archive/" + commit + ".zip"
  53. log.Infof("installing %v @ %v\n", plugin.Id, version)
  54. log.Infof("from url: %v\n", downloadURL)
  55. log.Infof("on commit: %v\n", commit)
  56. log.Infof("into: %v\n", pluginFolder)
  57. err = downloadFile(plugin.Id, pluginFolder, downloadURL)
  58. if err == nil {
  59. log.Info("Installed %s successfully ✔\n", plugin.Id)
  60. }
  61. res := services.ReadPlugin(pluginFolder, pluginName)
  62. for _, v := range res.Dependency.Plugins {
  63. log.Infof("Depends on %s install!\n", v.Id)
  64. //Todo: uncomment this code once the repo is more correct.
  65. //InstallPlugin(v.Id, pluginFolder, "")
  66. }
  67. return err
  68. }
  69. func SelectVersion(plugin m.Plugin, version string) (m.Version, error) {
  70. if version == "" {
  71. return plugin.Versions[0], nil
  72. }
  73. for _, v := range plugin.Versions {
  74. if v.Version == version {
  75. return v, nil
  76. }
  77. }
  78. return m.Version{}, errors.New("Could not find the version your looking for")
  79. }
  80. func RemoveGitBuildFromname(pluginname, filename string) string {
  81. r := regexp.MustCompile("^[a-zA-Z0-9_.-]*/")
  82. return r.ReplaceAllString(filename, pluginname+"/")
  83. }
  84. func downloadFile(pluginName, filepath, url string) (err error) {
  85. resp, err := http.Get(url)
  86. if err != nil {
  87. return err
  88. }
  89. defer resp.Body.Close()
  90. body, err := ioutil.ReadAll(resp.Body)
  91. if err != nil {
  92. return err
  93. }
  94. r, err := zip.NewReader(bytes.NewReader(body), resp.ContentLength)
  95. if err != nil {
  96. return err
  97. }
  98. for _, zf := range r.File {
  99. newfile := path.Join(filepath, RemoveGitBuildFromname(pluginName, zf.Name))
  100. if zf.FileInfo().IsDir() {
  101. os.Mkdir(newfile, 0777)
  102. } else {
  103. dst, err := os.Create(newfile)
  104. if err != nil {
  105. log.Errorf("%v", err)
  106. }
  107. defer dst.Close()
  108. src, err := zf.Open()
  109. if err != nil {
  110. log.Errorf("%v", err)
  111. }
  112. defer src.Close()
  113. io.Copy(dst, src)
  114. }
  115. }
  116. return nil
  117. }