aboutsummaryrefslogblamecommitdiffstats
path: root/repository.go
blob: 676c7b818ac41bd22ff4a9fef05b2041fff01f3b (plain) (tree)
1
2
3
4
5
6
7
8
9


           
                
 

                                                 
                                                     
                                                 
                                           

 
     
                                                          
                                                        

 
                                                                     

                                  
                            

 



                                                                           
 
                                                                                   



                                                                                 






                               

                                                                
                                     

 
                                   
                                                             


                                            
 
                                                                  







                                               










                                                        


                          
 




                                                                    
                       


                          




                                                             
 
 

                                                                                              
                       
                               

         








                                                     
         
 






















                                                                                      
                                                                
                                    
                                  


                                        

                          

 
                                                                       

                                     

                                                                    

         
                                                               


                          

                                                                

 































                                                                                                 






































                                                                             
                                                                
                                                         
                                  


                                        

                          

 
                                               
                                                           
                                              
                       

                                                     

                               






                                          
                                                     
                                                                




                                          

 

                                                       
                                              
                       

                                                     



                               

                                     

 

                                                       
                                              
                       

                                                     





                                     

 

                                                     
                                              
                       

                                                     

                               

         

                               



                                                                         
                                               
                                                             




                                       
 


                                                          
                                              
                       

                                                     




















                                                 
 

                                                      
                                                                       



                                                                                           
                     
                                                                          

         
                                               



                                                
                                             
                
 
package git

import (
	"errors"

	"gopkg.in/src-d/go-git.v4/clients/common"
	"gopkg.in/src-d/go-git.v4/core"
	"gopkg.in/src-d/go-git.v4/storage/filesystem"
	"gopkg.in/src-d/go-git.v4/storage/memory"
	"gopkg.in/src-d/go-git.v4/utils/fs"
)

var (
	ErrObjectNotFound = errors.New("object not found")
	ErrUnknownRemote  = errors.New("unknown remote")
)

// Repository giturl string, auth common.AuthMethod repository struct
type Repository struct {
	Remotes map[string]*Remote
	s       core.Storage
}

// NewMemoryRepository creates a new repository, backed by a memory.Storage
func NewMemoryRepository() (*Repository, error) {
	return NewRepository(memory.NewStorage())
}

// NewFilesystemRepository creates a new repository, backed by a filesystem.Storage
// based on a fs.OS, if you want to use a custom one you need to use the function
// NewRepository and build you filesystem.Storage
func NewFilesystemRepository(path string) (*Repository, error) {
	s, err := filesystem.NewStorage(fs.NewOS(), path)
	if err != nil {
		return nil, err
	}

	return NewRepository(s)
}

// NewRepository creates a new repository with the given Storage
func NewRepository(s core.Storage) (*Repository, error) {
	return &Repository{s: s}, nil
}

// Clone clones a remote repository
func (r *Repository) Clone(o *RepositoryCloneOptions) error {
	if err := o.Validate(); err != nil {
		return err
	}

	remote, err := r.createRemote(o.RemoteName, o.URL, o.Auth)
	if err != nil {
		return err
	}

	if err = remote.Connect(); err != nil {
		return err
	}

	var single core.ReferenceName
	if o.SingleBranch {
		single = o.ReferenceName
	}

	head, err := remote.Ref(o.ReferenceName, true)
	if err != nil {
		return err
	}

	refs, err := r.getRemoteRefences(remote, single)
	if err != nil {
		return err
	}

	err = remote.Fetch(r.s.ObjectStorage(), &RemoteFetchOptions{
		References: refs,
		Depth:      o.Depth,
	})

	if err != nil {
		return err
	}

	if err := r.createLocalReferences(head); err != nil {
		return err
	}

	return r.createRemoteReferences(remote, refs)
}

func (r *Repository) createRemote(name, url string, auth common.AuthMethod) (*Remote, error) {
	remote, err := NewAuthenticatedRemote(name, url, auth)
	if err != nil {
		return nil, err
	}

	r.Remotes = map[string]*Remote{name: remote}
	return remote, nil
}

func (r *Repository) getRemoteRefences(
	remote *Remote, single core.ReferenceName,
) ([]*core.Reference, error) {
	if single == "" {
		return r.getAllRemoteRefences(remote)
	}

	ref, err := remote.Ref(single, true)
	if err != nil {
		return nil, err
	}

	refs := []*core.Reference{ref}
	head, err := remote.Ref(core.HEAD, false)
	if err != nil {
		return nil, err
	}

	if head.Target() == ref.Name() {
		refs = append(refs, head)
	}

	return refs, nil
}

func (r *Repository) getAllRemoteRefences(remote *Remote) ([]*core.Reference, error) {
	var refs []*core.Reference
	i := remote.Refs()
	defer i.Close()

	return refs, i.ForEach(func(ref *core.Reference) error {
		if !ref.IsBranch() {
			return nil
		}

		refs = append(refs, ref)
		return nil
	})
}

func (r *Repository) createLocalReferences(ref *core.Reference) error {
	if !ref.IsBranch() {
		// detached HEAD mode
		head := core.NewHashReference(core.HEAD, ref.Hash())
		return r.s.ReferenceStorage().Set(head)
	}

	if err := r.s.ReferenceStorage().Set(ref); err != nil {
		return err
	}

	head := core.NewSymbolicReference(core.HEAD, ref.Name())
	return r.s.ReferenceStorage().Set(head)
}

func (r *Repository) createRemoteReferences(remote *Remote, remoteRefs []*core.Reference) error {
	for _, ref := range remoteRefs {
		if err := r.createRemoteReference(remote, ref); err != nil {
			return err
		}
	}

	return nil
}

func (r *Repository) createRemoteReference(remote *Remote, ref *core.Reference) error {
	name := ref.Name().AsRemote(remote.Name)

	var n *core.Reference
	switch ref.Type() {
	case core.HashReference:
		n = core.NewHashReference(name, ref.Hash())
	case core.SymbolicReference:
		n = core.NewSymbolicReference(name, ref.Target().AsRemote(remote.Name))
		target, err := remote.Ref(ref.Target(), false)
		if err != nil {
			return err
		}

		if err := r.createRemoteReference(remote, target); err != nil {
			return err
		}
	}

	return r.s.ReferenceStorage().Set(n)
}

// Pull incorporates changes from a remote repository into the current branch
func (r *Repository) Pull(o *RepositoryPullOptions) error {
	if err := o.Validate(); err != nil {
		return err
	}

	remote, ok := r.Remotes[o.RemoteName]
	if !ok {
		return ErrUnknownRemote
	}

	head, err := remote.Ref(o.ReferenceName, true)
	if err != nil {
		return err
	}

	refs, err := r.getLocalReferences()
	if err != nil {
		return err
	}

	err = remote.Fetch(r.s.ObjectStorage(), &RemoteFetchOptions{
		References:      []*core.Reference{head},
		LocalReferences: refs,
		Depth:           o.Depth,
	})

	if err != nil {
		return err
	}

	return r.createLocalReferences(head)
}

func (r *Repository) getLocalReferences() ([]*core.Reference, error) {
	var refs []*core.Reference
	i := r.Refs()
	defer i.Close()

	return refs, i.ForEach(func(ref *core.Reference) error {
		if ref.Type() == core.SymbolicReference {
			return nil
		}

		refs = append(refs, ref)
		return nil
	})
}

// Commit return the commit with the given hash
func (r *Repository) Commit(h core.Hash) (*Commit, error) {
	obj, err := r.s.ObjectStorage().Get(h)
	if err != nil {
		if err == core.ErrObjectNotFound {
			return nil, ErrObjectNotFound
		}
		return nil, err
	}

	commit := &Commit{r: r}
	return commit, commit.Decode(obj)
}

// Commits decode the objects into commits
func (r *Repository) Commits() (*CommitIter, error) {
	iter, err := r.s.ObjectStorage().Iter(core.CommitObject)
	if err != nil {
		return nil, err
	}

	return NewCommitIter(r, iter), nil
}

// Tree return the tree with the given hash
func (r *Repository) Tree(h core.Hash) (*Tree, error) {
	obj, err := r.s.ObjectStorage().Get(h)
	if err != nil {
		if err == core.ErrObjectNotFound {
			return nil, ErrObjectNotFound
		}
		return nil, err
	}

	tree := &Tree{r: r}
	return tree, tree.Decode(obj)
}

// Blob returns the blob with the given hash
func (r *Repository) Blob(h core.Hash) (*Blob, error) {
	obj, err := r.s.ObjectStorage().Get(h)
	if err != nil {
		if err == core.ErrObjectNotFound {
			return nil, ErrObjectNotFound
		}
		return nil, err
	}

	blob := &Blob{}
	return blob, blob.Decode(obj)
}

// Tag returns a tag with the given hash.
func (r *Repository) Tag(h core.Hash) (*Tag, error) {
	obj, err := r.s.ObjectStorage().Get(h)
	if err != nil {
		if err == core.ErrObjectNotFound {
			return nil, ErrObjectNotFound
		}
		return nil, err
	}

	t := &Tag{r: r}
	return t, t.Decode(obj)
}

// Tags returns a TagIter that can step through all of the annotated tags
// in the repository.
func (r *Repository) Tags() (*TagIter, error) {
	iter, err := r.s.ObjectStorage().Iter(core.TagObject)
	if err != nil {
		return nil, err
	}

	return NewTagIter(r, iter), nil
}

// Object returns an object with the given hash.
func (r *Repository) Object(h core.Hash) (Object, error) {
	obj, err := r.s.ObjectStorage().Get(h)
	if err != nil {
		if err == core.ErrObjectNotFound {
			return nil, ErrObjectNotFound
		}
		return nil, err
	}

	switch obj.Type() {
	case core.CommitObject:
		commit := &Commit{r: r}
		return commit, commit.Decode(obj)
	case core.TreeObject:
		tree := &Tree{r: r}
		return tree, tree.Decode(obj)
	case core.BlobObject:
		blob := &Blob{}
		return blob, blob.Decode(obj)
	case core.TagObject:
		tag := &Tag{r: r}
		return tag, tag.Decode(obj)
	default:
		return nil, core.ErrInvalidType
	}
}

// Head returns the reference where HEAD is pointing
func (r *Repository) Head() (*core.Reference, error) {
	return core.ResolveReference(r.s.ReferenceStorage(), core.HEAD)
}

// Ref returns the Hash pointing the given refName
func (r *Repository) Ref(name core.ReferenceName, resolved bool) (*core.Reference, error) {
	if resolved {
		return core.ResolveReference(r.s.ReferenceStorage(), name)
	}

	return r.s.ReferenceStorage().Get(name)
}

// Refs returns a map with all the References
func (r *Repository) Refs() core.ReferenceIter {
	i, _ := r.s.ReferenceStorage().Iter()
	return i
}