Tool to help you manage your Grafana dashboards using Git.

git.go 5.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. package git
  2. import (
  3. "io/ioutil"
  4. "os"
  5. "strings"
  6. "config"
  7. "github.com/sirupsen/logrus"
  8. "golang.org/x/crypto/ssh"
  9. gogit "gopkg.in/src-d/go-git.v4"
  10. "gopkg.in/src-d/go-git.v4/plumbing/transport"
  11. gitssh "gopkg.in/src-d/go-git.v4/plumbing/transport/ssh"
  12. )
  13. // Sync synchronises a Git repository using a given configuration. "synchronises"
  14. // means that, if the repo from the configuration isn't already cloned in the
  15. // directory specified in the configuration, it will clone the repository,
  16. // else it will simply pull it in order to be up to date with the remote.
  17. // Returns the go-git representation of the repository.
  18. // Returns an error if there was an issue loading the SSH private key, checking
  19. // whether the clone path already exists, or synchronising the repo with the
  20. // remote.
  21. func Sync(cfg config.GitSettings) (r *gogit.Repository, err error) {
  22. // Generate an authentication structure instance from the user and private
  23. // key
  24. auth, err := getAuth(cfg.User, cfg.PrivateKeyPath)
  25. if err != nil {
  26. return
  27. }
  28. // Check whether the clone path already exists
  29. exists, err := dirExists(cfg.ClonePath)
  30. if err != nil {
  31. return
  32. }
  33. logrus.WithFields(logrus.Fields{
  34. "repo": cfg.User + "@" + cfg.URL,
  35. "clone_path": cfg.ClonePath,
  36. "pull": exists,
  37. }).Info("Synchronising the Git repository with the remote")
  38. // If the clone path already exists, pull from the remote, else clone it.
  39. if exists {
  40. r, err = pull(cfg.ClonePath, auth)
  41. } else {
  42. r, err = clone(cfg.URL, cfg.ClonePath, auth)
  43. }
  44. return
  45. }
  46. // getAuth returns the authentication structure instance needed to authenticate
  47. // on the remote, using a given user and private key path.
  48. // Returns an error if there was an issue reading the private key file or
  49. // parsing it.
  50. func getAuth(user string, privateKeyPath string) (*gitssh.PublicKeys, error) {
  51. privateKey, err := ioutil.ReadFile(privateKeyPath)
  52. if err != nil {
  53. return nil, err
  54. }
  55. signer, err := ssh.ParsePrivateKey(privateKey)
  56. if err != nil {
  57. return nil, err
  58. }
  59. return &gitssh.PublicKeys{User: user, Signer: signer}, nil
  60. }
  61. // clone clones a Git repository into a given path, using a given auth.
  62. // Returns the go-git representation of the Git repository.
  63. // Returns an error if there was an issue cloning the repository.
  64. func clone(repo string, clonePath string, auth *gitssh.PublicKeys) (*gogit.Repository, error) {
  65. return gogit.PlainClone(clonePath, false, &gogit.CloneOptions{
  66. URL: repo,
  67. Auth: auth,
  68. })
  69. }
  70. // pull opens the repository located at a given path, and pulls it from the
  71. // remote using a given auth, in order to be up to date with the remote.
  72. // Returns with the go-git representation of the repository.
  73. // Returns an error if there was an issue opening the repo, getting its work
  74. // tree or pulling from the remote. In the latter case, if the error is a known
  75. // non-error, doesn't return any error.
  76. func pull(clonePath string, auth *gitssh.PublicKeys) (*gogit.Repository, error) {
  77. // Open the repository
  78. r, err := gogit.PlainOpen(clonePath)
  79. if err != nil {
  80. return nil, err
  81. }
  82. // Get its worktree
  83. w, err := r.Worktree()
  84. if err != nil {
  85. return nil, err
  86. }
  87. // Pull from remote
  88. if err = w.Pull(&gogit.PullOptions{
  89. RemoteName: "origin",
  90. Auth: auth,
  91. }); err != nil {
  92. // Check error against known non-errors
  93. err = checkRemoteErrors(err, logrus.Fields{
  94. "clone_path": clonePath,
  95. "error": err,
  96. })
  97. }
  98. return r, err
  99. }
  100. // dirExists is a snippet checking if a directory exists on the disk.
  101. // Returns with a boolean set to true if the directory exists, false if not.
  102. // Returns with an error if there was an issue checking the directory's
  103. // existence.
  104. func dirExists(path string) (bool, error) {
  105. _, err := os.Stat(path)
  106. if os.IsNotExist(err) {
  107. return false, nil
  108. }
  109. return true, err
  110. }
  111. // Push uses a given repository and configuration to push the local history of
  112. // the said repository to the remote, using an authentication structure instance
  113. // created from the configuration to authenticate on the remote.
  114. // Returns with an error if there was an issue creating the authentication
  115. // structure instance or pushing to the remote. In the latter case, if the error
  116. // is a known non-error, doesn't return any error.
  117. func Push(r *gogit.Repository, cfg config.GitSettings) error {
  118. // Get the authentication structure instance
  119. auth, err := getAuth(cfg.User, cfg.PrivateKeyPath)
  120. if err != nil {
  121. return err
  122. }
  123. logrus.WithFields(logrus.Fields{
  124. "repo": cfg.User + "@" + cfg.URL,
  125. "clone_path": cfg.ClonePath,
  126. }).Info("Pushing to the remote")
  127. // Push to remote
  128. if err = r.Push(&gogit.PushOptions{
  129. Auth: auth,
  130. }); err != nil {
  131. // Check error against known non-errors
  132. err = checkRemoteErrors(err, logrus.Fields{
  133. "repo": cfg.User + "@" + cfg.URL,
  134. "clone_path": cfg.ClonePath,
  135. "error": err,
  136. })
  137. }
  138. return err
  139. }
  140. // processRemoteErrors checks an error against known non-errors returned when
  141. // communicating with the remote. If the error is a non-error, returns nil and
  142. // logs it with the provided fields. If not, returns the error.
  143. // Current known non-errors are "already up to date" and "remote repository is
  144. // empty".
  145. func checkRemoteErrors(err error, logFields logrus.Fields) error {
  146. var nonError bool
  147. // Check against known non-errors
  148. switch err {
  149. case gogit.NoErrAlreadyUpToDate:
  150. nonError = true
  151. break
  152. case transport.ErrEmptyRemoteRepository:
  153. nonError = true
  154. break
  155. default:
  156. nonError = false
  157. break
  158. }
  159. // Log non-error
  160. if nonError {
  161. logrus.WithFields(logFields).Warn("Caught specific non-error")
  162. return nil
  163. }
  164. return err
  165. }