git2go/blob.go

152 lines
3.3 KiB
Go
Raw Permalink Normal View History

2013-03-05 13:53:04 -06:00
package git
/*
#include <git2.h>
#include <string.h>
2016-04-21 09:34:51 -05:00
int _go_git_writestream_write(git_writestream *stream, const char *buffer, size_t len);
void _go_git_writestream_free(git_writestream *stream);
2013-03-05 13:53:04 -06:00
*/
import "C"
import (
2016-04-21 09:34:51 -05:00
"reflect"
"runtime"
2013-03-05 13:53:04 -06:00
"unsafe"
)
type Blob struct {
doNotCompare
Object
cast_ptr *C.git_blob
2013-04-16 16:04:35 -05:00
}
func (b *Blob) AsObject() *Object {
return &b.Object
}
func (v *Blob) Size() int64 {
ret := int64(C.git_blob_rawsize(v.cast_ptr))
runtime.KeepAlive(v)
return ret
}
func (v *Blob) Contents() []byte {
size := C.int(C.git_blob_rawsize(v.cast_ptr))
buffer := unsafe.Pointer(C.git_blob_rawcontent(v.cast_ptr))
goBytes := C.GoBytes(buffer, size)
runtime.KeepAlive(v)
return goBytes
2013-03-05 13:53:04 -06:00
}
func (v *Blob) IsBinary() bool {
ret := C.git_blob_is_binary(v.cast_ptr) == 1
runtime.KeepAlive(v)
return ret
}
func (repo *Repository) CreateBlobFromBuffer(data []byte) (*Oid, error) {
runtime.LockOSThread()
defer runtime.UnlockOSThread()
var id C.git_oid
var size C.size_t
// Go 1.6 added some increased checking of passing pointer to
2017-07-11 22:52:13 -05:00
// C, but its check depends on its expectations of what we
// pass to the C function, so unless we take the address of
// its contents at the call site itself, it can fail when
// 'data' is a slice of a slice.
//
// When we're given an empty slice, create a dummy one where 0
// isn't out of bounds.
if len(data) > 0 {
size = C.size_t(len(data))
} else {
data = []byte{0}
size = C.size_t(0)
}
ecode := C.git_blob_create_from_buffer(&id, repo.ptr, unsafe.Pointer(&data[0]), size)
runtime.KeepAlive(repo)
if ecode < 0 {
return nil, MakeGitError(ecode)
}
return newOidFromC(&id), nil
}
2016-04-21 09:34:51 -05:00
func (repo *Repository) CreateFromStream(hintPath string) (*BlobWriteStream, error) {
var chintPath *C.char = nil
2016-04-21 09:34:51 -05:00
var stream *C.git_writestream
if len(hintPath) > 0 {
chintPath = C.CString(hintPath)
defer C.free(unsafe.Pointer(chintPath))
}
2016-04-21 09:34:51 -05:00
runtime.LockOSThread()
defer runtime.UnlockOSThread()
ecode := C.git_blob_create_from_stream(&stream, repo.ptr, chintPath)
2016-04-21 09:34:51 -05:00
if ecode < 0 {
return nil, MakeGitError(ecode)
}
return newBlobWriteStreamFromC(stream, repo), nil
2016-04-21 09:34:51 -05:00
}
type BlobWriteStream struct {
doNotCompare
ptr *C.git_writestream
repo *Repository
2016-04-21 09:34:51 -05:00
}
func newBlobWriteStreamFromC(ptr *C.git_writestream, repo *Repository) *BlobWriteStream {
2016-04-21 09:34:51 -05:00
stream := &BlobWriteStream{
ptr: ptr,
repo: repo,
2016-04-21 09:34:51 -05:00
}
runtime.SetFinalizer(stream, (*BlobWriteStream).Free)
return stream
}
// Implement io.Writer
func (stream *BlobWriteStream) Write(p []byte) (int, error) {
header := (*reflect.SliceHeader)(unsafe.Pointer(&p))
ptr := (*C.char)(unsafe.Pointer(header.Data))
size := C.size_t(header.Len)
2016-04-21 09:34:51 -05:00
runtime.LockOSThread()
defer runtime.UnlockOSThread()
ecode := C._go_git_writestream_write(stream.ptr, ptr, size)
runtime.KeepAlive(stream)
2016-04-21 09:34:51 -05:00
if ecode < 0 {
return 0, MakeGitError(ecode)
}
2016-04-21 09:34:51 -05:00
return len(p), nil
}
func (stream *BlobWriteStream) Free() {
runtime.SetFinalizer(stream, nil)
C._go_git_writestream_free(stream.ptr)
}
func (stream *BlobWriteStream) Commit() (*Oid, error) {
oid := C.git_oid{}
runtime.LockOSThread()
defer runtime.UnlockOSThread()
ecode := C.git_blob_create_from_stream_commit(&oid, stream.ptr)
runtime.KeepAlive(stream)
if ecode < 0 {
return nil, MakeGitError(ecode)
}
2016-04-21 09:34:51 -05:00
return newOidFromC(&oid), nil
}