2013-03-05 13:53:04 -06:00
|
|
|
package git
|
|
|
|
|
|
|
|
/*
|
|
|
|
#cgo pkg-config: libgit2
|
|
|
|
#include <git2.h>
|
|
|
|
#include <git2/errors.h>
|
|
|
|
*/
|
|
|
|
import "C"
|
|
|
|
import (
|
2013-04-26 14:12:27 -05:00
|
|
|
"bytes"
|
2014-03-18 21:39:34 -05:00
|
|
|
"encoding/hex"
|
2013-05-21 08:14:26 -05:00
|
|
|
"errors"
|
2013-09-18 02:23:47 -05:00
|
|
|
"runtime"
|
2013-05-14 14:34:07 -05:00
|
|
|
"strings"
|
2014-02-26 12:41:20 -06:00
|
|
|
"unsafe"
|
2013-03-05 13:53:04 -06:00
|
|
|
)
|
|
|
|
|
2014-04-02 12:31:48 -05:00
|
|
|
type ErrorClass int
|
|
|
|
|
2013-03-05 13:53:04 -06:00
|
|
|
const (
|
2014-04-02 12:31:48 -05:00
|
|
|
ErrClassNone ErrorClass = C.GITERR_NONE
|
|
|
|
ErrClassNoMemory = C.GITERR_NOMEMORY
|
|
|
|
ErrClassOs = C.GITERR_OS
|
|
|
|
ErrClassInvalid = C.GITERR_INVALID
|
|
|
|
ErrClassReference = C.GITERR_REFERENCE
|
|
|
|
ErrClassZlib = C.GITERR_ZLIB
|
|
|
|
ErrClassRepository = C.GITERR_REPOSITORY
|
|
|
|
ErrClassConfig = C.GITERR_CONFIG
|
|
|
|
ErrClassRegex = C.GITERR_REGEX
|
|
|
|
ErrClassOdb = C.GITERR_ODB
|
|
|
|
ErrClassIndex = C.GITERR_INDEX
|
|
|
|
ErrClassObject = C.GITERR_OBJECT
|
|
|
|
ErrClassNet = C.GITERR_NET
|
|
|
|
ErrClassTag = C.GITERR_TAG
|
|
|
|
ErrClassTree = C.GITERR_TREE
|
|
|
|
ErrClassIndexer = C.GITERR_INDEXER
|
|
|
|
ErrClassSSL = C.GITERR_SSL
|
|
|
|
ErrClassSubmodule = C.GITERR_SUBMODULE
|
|
|
|
ErrClassThread = C.GITERR_THREAD
|
|
|
|
ErrClassStash = C.GITERR_STASH
|
|
|
|
ErrClassCheckout = C.GITERR_CHECKOUT
|
|
|
|
ErrClassFetchHead = C.GITERR_FETCHHEAD
|
|
|
|
ErrClassMerge = C.GITERR_MERGE
|
|
|
|
ErrClassSsh = C.GITERR_SSH
|
|
|
|
ErrClassFilter = C.GITERR_FILTER
|
|
|
|
ErrClassRevert = C.GITERR_REVERT
|
|
|
|
ErrClassCallback = C.GITERR_CALLBACK
|
2013-03-05 13:53:04 -06:00
|
|
|
)
|
|
|
|
|
2014-04-02 12:31:48 -05:00
|
|
|
type ErrorCode int
|
|
|
|
|
|
|
|
const (
|
|
|
|
ErrOk ErrorCode = C.GIT_OK /*< No error */
|
|
|
|
|
|
|
|
ErrGeneric = C.GIT_ERROR /*< Generic error */
|
|
|
|
ErrNotFound = C.GIT_ENOTFOUND /*< Requested object could not be found */
|
|
|
|
ErrExists = C.GIT_EEXISTS /*< Object exists preventing operation */
|
|
|
|
ErrAmbigious = C.GIT_EAMBIGUOUS /*< More than one object matches */
|
|
|
|
ErrBuffs = C.GIT_EBUFS /*< Output buffer too short to hold data */
|
|
|
|
|
|
|
|
/* GIT_EUSER is a special error that is never generated by libgit2
|
|
|
|
* code. You can return it from a callback (e.g to stop an iteration)
|
|
|
|
* to know that it was generated by the callback and not by libgit2.
|
|
|
|
*/
|
|
|
|
ErrUser = C.GIT_EUSER
|
|
|
|
|
|
|
|
ErrBareRepo = C.GIT_EBAREREPO /*< Operation not allowed on bare repository */
|
|
|
|
ErrUnbornBranch = C.GIT_EUNBORNBRANCH /*< HEAD refers to branch with no commits */
|
|
|
|
ErrUnmerged = C.GIT_EUNMERGED /*< Merge in progress prevented operation */
|
|
|
|
ErrNonFastForward = C.GIT_ENONFASTFORWARD /*< Reference was not fast-forwardable */
|
|
|
|
ErrInvalidSpec = C.GIT_EINVALIDSPEC /*< Name/ref spec was not in a valid format */
|
|
|
|
ErrMergeConflict = C.GIT_EMERGECONFLICT /*< Merge conflicts prevented operation */
|
|
|
|
ErrLocked = C.GIT_ELOCKED /*< Lock file prevented operation */
|
|
|
|
ErrModified = C.GIT_EMODIFIED /*< Reference value does not match expected */
|
|
|
|
|
|
|
|
ErrPassthrough = C.GIT_PASSTHROUGH /*< Internal only */
|
|
|
|
ErrIterOver = C.GIT_ITEROVER /*< Signals end of iteration with iterator */
|
2013-05-21 08:14:26 -05:00
|
|
|
)
|
|
|
|
|
2013-03-05 13:53:04 -06:00
|
|
|
func init() {
|
|
|
|
C.git_threads_init()
|
|
|
|
}
|
|
|
|
|
2014-03-18 21:11:41 -05:00
|
|
|
// Oid represents the id for a Git object.
|
2014-03-17 22:54:40 -05:00
|
|
|
type Oid [20]byte
|
2013-03-05 13:53:04 -06:00
|
|
|
|
|
|
|
func newOidFromC(coid *C.git_oid) *Oid {
|
2013-03-08 09:03:49 -06:00
|
|
|
if coid == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-03-05 13:53:04 -06:00
|
|
|
oid := new(Oid)
|
2014-03-17 22:54:40 -05:00
|
|
|
copy(oid[0:20], C.GoBytes(unsafe.Pointer(coid), 20))
|
2013-03-05 13:53:04 -06:00
|
|
|
return oid
|
|
|
|
}
|
|
|
|
|
2014-03-17 23:04:26 -05:00
|
|
|
func NewOidFromBytes(b []byte) *Oid {
|
2013-03-05 13:53:04 -06:00
|
|
|
oid := new(Oid)
|
2014-03-17 22:54:40 -05:00
|
|
|
copy(oid[0:20], b[0:20])
|
2013-03-05 13:53:04 -06:00
|
|
|
return oid
|
|
|
|
}
|
|
|
|
|
|
|
|
func (oid *Oid) toC() *C.git_oid {
|
2014-03-17 22:54:40 -05:00
|
|
|
return (*C.git_oid)(unsafe.Pointer(oid))
|
2013-03-05 13:53:04 -06:00
|
|
|
}
|
|
|
|
|
2014-03-17 23:04:26 -05:00
|
|
|
func NewOid(s string) (*Oid, error) {
|
2014-03-18 21:39:34 -05:00
|
|
|
if len(s) > C.GIT_OID_HEXSZ {
|
|
|
|
return nil, errors.New("string is too long for oid")
|
|
|
|
}
|
2013-03-05 13:53:04 -06:00
|
|
|
|
2014-03-18 21:39:34 -05:00
|
|
|
o := new(Oid)
|
2013-09-18 02:23:47 -05:00
|
|
|
|
2014-03-18 21:39:34 -05:00
|
|
|
slice, error := hex.DecodeString(s)
|
|
|
|
if error != nil {
|
|
|
|
return nil, error
|
2013-03-05 13:53:04 -06:00
|
|
|
}
|
|
|
|
|
2014-04-02 12:31:48 -05:00
|
|
|
if len(slice) != 20 {
|
|
|
|
return nil, &GitError{"Invalid Oid", ErrClassNone, ErrGeneric}
|
|
|
|
}
|
|
|
|
|
2014-03-18 21:39:34 -05:00
|
|
|
copy(o[:], slice[:20])
|
2013-03-05 13:53:04 -06:00
|
|
|
return o, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (oid *Oid) String() string {
|
2014-03-18 21:39:34 -05:00
|
|
|
return hex.EncodeToString(oid[:])
|
2013-03-05 13:53:04 -06:00
|
|
|
}
|
|
|
|
|
2013-04-26 14:12:27 -05:00
|
|
|
func (oid *Oid) Cmp(oid2 *Oid) int {
|
2014-03-17 22:54:40 -05:00
|
|
|
return bytes.Compare(oid[:], oid2[:])
|
2013-04-26 14:12:27 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (oid *Oid) Copy() *Oid {
|
|
|
|
ret := new(Oid)
|
2014-03-17 22:54:40 -05:00
|
|
|
copy(ret[:], oid[:])
|
2013-04-26 14:12:27 -05:00
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
|
|
|
func (oid *Oid) Equal(oid2 *Oid) bool {
|
2014-03-17 22:54:40 -05:00
|
|
|
return bytes.Equal(oid[:], oid2[:])
|
2013-04-26 14:12:27 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func (oid *Oid) IsZero() bool {
|
2014-03-17 22:54:40 -05:00
|
|
|
for _, a := range oid {
|
2014-03-18 21:51:59 -05:00
|
|
|
if a != 0 {
|
2013-04-26 14:12:27 -05:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func (oid *Oid) NCmp(oid2 *Oid, n uint) int {
|
2014-03-17 22:54:40 -05:00
|
|
|
return bytes.Compare(oid[:n], oid2[:n])
|
2013-04-26 14:12:27 -05:00
|
|
|
}
|
|
|
|
|
2013-05-21 16:03:11 -05:00
|
|
|
func ShortenOids(ids []*Oid, minlen int) (int, error) {
|
|
|
|
shorten := C.git_oid_shorten_new(C.size_t(minlen))
|
|
|
|
if shorten == nil {
|
|
|
|
panic("Out of memory")
|
|
|
|
}
|
|
|
|
defer C.git_oid_shorten_free(shorten)
|
|
|
|
|
|
|
|
var ret C.int
|
2013-09-18 02:23:47 -05:00
|
|
|
|
|
|
|
runtime.LockOSThread()
|
|
|
|
defer runtime.UnlockOSThread()
|
|
|
|
|
2013-05-21 16:03:11 -05:00
|
|
|
for _, id := range ids {
|
|
|
|
buf := make([]byte, 41)
|
|
|
|
C.git_oid_fmt((*C.char)(unsafe.Pointer(&buf[0])), id.toC())
|
|
|
|
buf[40] = 0
|
|
|
|
ret = C.git_oid_shorten_add(shorten, (*C.char)(unsafe.Pointer(&buf[0])))
|
|
|
|
if ret < 0 {
|
2013-07-07 09:43:44 -05:00
|
|
|
return int(ret), MakeGitError(ret)
|
2013-05-21 16:03:11 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return int(ret), nil
|
|
|
|
}
|
|
|
|
|
2013-03-05 13:53:04 -06:00
|
|
|
type GitError struct {
|
2014-04-02 12:31:48 -05:00
|
|
|
Message string
|
|
|
|
Class ErrorClass
|
|
|
|
Code ErrorCode
|
2013-03-05 13:53:04 -06:00
|
|
|
}
|
|
|
|
|
2014-01-29 17:01:26 -06:00
|
|
|
func (e GitError) Error() string {
|
2013-03-05 13:53:04 -06:00
|
|
|
return e.Message
|
|
|
|
}
|
|
|
|
|
2014-04-02 12:31:48 -05:00
|
|
|
func IsErrorClass(err error, c ErrorClass) bool {
|
|
|
|
|
|
|
|
if err == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if gitError, ok := err.(*GitError); ok {
|
|
|
|
return gitError.Class == c
|
|
|
|
}
|
|
|
|
return false
|
2013-07-07 09:43:44 -05:00
|
|
|
}
|
|
|
|
|
2014-04-02 12:31:48 -05:00
|
|
|
func IsErrorCode(err error, c ErrorCode) bool {
|
|
|
|
if err == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if gitError, ok := err.(*GitError); ok {
|
|
|
|
return gitError.Code == c
|
|
|
|
}
|
|
|
|
return false
|
2013-07-07 09:43:44 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func MakeGitError(errorCode C.int) error {
|
2014-04-02 12:31:48 -05:00
|
|
|
|
|
|
|
var errMessage string
|
|
|
|
var errClass ErrorClass
|
|
|
|
if errorCode != ErrIterOver {
|
|
|
|
err := C.giterr_last()
|
|
|
|
if err != nil {
|
|
|
|
errMessage = C.GoString(err.message)
|
|
|
|
errClass = ErrorClass(err.klass)
|
|
|
|
} else {
|
|
|
|
errClass = ErrClassInvalid
|
|
|
|
}
|
2013-05-21 04:51:31 -05:00
|
|
|
}
|
2014-04-02 12:31:48 -05:00
|
|
|
return &GitError{errMessage, errClass, ErrorCode(errorCode)}
|
2013-03-05 13:53:04 -06:00
|
|
|
}
|
|
|
|
|
2014-03-24 18:12:32 -05:00
|
|
|
func MakeGitError2(err int) error {
|
|
|
|
return MakeGitError(C.int(err))
|
|
|
|
}
|
|
|
|
|
2013-03-05 13:53:04 -06:00
|
|
|
func cbool(b bool) C.int {
|
2014-01-29 17:01:26 -06:00
|
|
|
if b {
|
2013-03-05 13:53:04 -06:00
|
|
|
return C.int(1)
|
|
|
|
}
|
|
|
|
return C.int(0)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ucbool(b bool) C.uint {
|
2014-01-29 17:01:26 -06:00
|
|
|
if b {
|
2013-03-05 13:53:04 -06:00
|
|
|
return C.uint(1)
|
|
|
|
}
|
|
|
|
return C.uint(0)
|
|
|
|
}
|
2013-05-14 14:34:07 -05:00
|
|
|
|
|
|
|
func Discover(start string, across_fs bool, ceiling_dirs []string) (string, error) {
|
|
|
|
ceildirs := C.CString(strings.Join(ceiling_dirs, string(C.GIT_PATH_LIST_SEPARATOR)))
|
|
|
|
defer C.free(unsafe.Pointer(ceildirs))
|
|
|
|
|
|
|
|
cstart := C.CString(start)
|
|
|
|
defer C.free(unsafe.Pointer(cstart))
|
|
|
|
|
2014-02-23 08:31:22 -06:00
|
|
|
var buf C.git_buf
|
|
|
|
defer C.git_buf_free(&buf)
|
2013-05-14 14:34:07 -05:00
|
|
|
|
2013-09-18 02:23:47 -05:00
|
|
|
runtime.LockOSThread()
|
|
|
|
defer runtime.UnlockOSThread()
|
|
|
|
|
2014-02-23 08:31:22 -06:00
|
|
|
ret := C.git_repository_discover(&buf, cstart, cbool(across_fs), ceildirs)
|
|
|
|
if ret < 0 {
|
2013-07-07 09:43:44 -05:00
|
|
|
return "", MakeGitError(ret)
|
2013-05-14 14:34:07 -05:00
|
|
|
}
|
|
|
|
|
2014-02-23 08:31:22 -06:00
|
|
|
return C.GoString(buf.ptr), nil
|
2013-05-14 14:34:07 -05:00
|
|
|
}
|