Tool to help you manage your Grafana dashboards using Git.

git.go 4.8KB

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