123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164 |
- package git
-
- import (
- "io/ioutil"
- "os"
- "strings"
-
- "config"
-
- "golang.org/x/crypto/ssh"
- gogit "gopkg.in/src-d/go-git.v4"
- gitssh "gopkg.in/src-d/go-git.v4/plumbing/transport/ssh"
- )
-
- // Sync synchronises a Git repository using a given configuration. "synchronises"
- // means that, if the repo from the configuration isn't already cloned in the
- // directory specified in the configuration, it will clone the repository,
- // else it will simply pull it in order to be up to date with the remote.
- // Returns the go-git representation of the repository.
- // Returns an error if there was an issue loading the SSH private key, checking
- // whether the clone path already exists, or synchronising the repo with the
- // remote.
- func Sync(cfg config.GitSettings) (r *gogit.Repository, err error) {
- // Generate an authentication structure instance from the user and private
- // key
- auth, err := getAuth(cfg.User, cfg.PrivateKeyPath)
- if err != nil {
- return
- }
-
- // Check whether the clone path already exists
- exists, err := dirExists(cfg.ClonePath)
- if err != nil {
- return
- }
-
- // If the clone path already exists, pull from the remote, else clone it.
- if exists {
- r, err = pull(cfg.ClonePath, auth)
- } else {
- r, err = clone(cfg.URL, cfg.ClonePath, auth)
- }
-
- return
- }
-
- // getAuth returns the authentication structure instance needed to authenticate
- // on the remote, using a given user and private key path.
- // Returns an error if there was an issue reading the private key file or
- // parsing it.
- func getAuth(user string, privateKeyPath string) (*gitssh.PublicKeys, error) {
- privateKey, err := ioutil.ReadFile(privateKeyPath)
- if err != nil {
- return nil, err
- }
-
- signer, err := ssh.ParsePrivateKey(privateKey)
- if err != nil {
- return nil, err
- }
-
- return &gitssh.PublicKeys{User: "git", Signer: signer}, nil
- }
-
- // clone clones a Git repository into a given path, using a given auth.
- // Returns the go-git representation of the Git repository.
- // Returns an error if there was an issue cloning the repository.
- func clone(repo string, clonePath string, auth *gitssh.PublicKeys) (*gogit.Repository, error) {
- return gogit.PlainClone(clonePath, false, &gogit.CloneOptions{
- URL: repo,
- Auth: auth,
- })
- }
-
- // pull opens the repository located at a given path, and pulls it from the
- // remote using a given auth, in order to be up to date with the remote.
- // Returns with the go-git representation of the repository.
- // Returns an error if there was an issue opening the repo, getting its work
- // tree or pulling from the remote. In the latter case, if the error is "already
- // up to date" or "non-fast-forward update", doesn't return any error.
- func pull(clonePath string, auth *gitssh.PublicKeys) (*gogit.Repository, error) {
- // Open the repository
- r, err := gogit.PlainOpen(clonePath)
- if err != nil {
- return nil, err
- }
-
- // Get its worktree
- w, err := r.Worktree()
- if err != nil {
- return nil, err
- }
-
- // Pull from remote
- err = w.Pull(&gogit.PullOptions{
- RemoteName: "origin",
- Auth: auth,
- })
-
- // Don't return with an error for "already up to date" or "non-fast-forward
- // update"
- if err != nil {
- if err == gogit.NoErrAlreadyUpToDate {
- return r, nil
- }
-
- // go-git doesn't have an error variable for "non-fast-forward update",
- // so this is the only way to detect it
- if strings.HasPrefix(err.Error(), "non-fast-forward update") {
- return r, nil
- }
- }
-
- return r, err
- }
-
- // dirExists is a snippet checking if a directory exists on the disk.
- // Returns with a boolean set to true if the directory exists, false if not.
- // Returns with an error if there was an issue checking the directory's
- // existence.
- func dirExists(path string) (bool, error) {
- _, err := os.Stat(path)
-
- if os.IsNotExist(err) {
- return false, nil
- }
-
- return true, err
- }
-
- // Push uses a given repository and configuration to push the local history of
- // the said repository to the remote, using an authentication structure instance
- // created from the configuration to authenticate on the remote.
- // Returns with an error if there was an issue creating the authentication
- // structure instance or pushing to the remote. In the latter case, if the error
- // is "already up to date" or "non-fast-forward update", doesn't return any error.
- func Push(r *gogit.Repository, cfg config.GitSettings) error {
- // Get the authentication structure instance
- auth, err := getAuth(cfg.User, cfg.PrivateKeyPath)
- if err != nil {
- return err
- }
-
- // Push to remote
- err = r.Push(&gogit.PushOptions{
- Auth: auth,
- })
-
- // Don't return with an error for "already up to date" or "non-fast-forward
- // update"
- if err != nil {
- if err == gogit.NoErrAlreadyUpToDate {
- return nil
- }
-
- // go-git doesn't have an error variable for "non-fast-forward update", so
- // this is the only way to detect it
- if strings.HasPrefix(err.Error(), "non-fast-forward update") {
- return nil
- }
- }
-
- return err
- }
|