1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
// Package otbuiltin contains all of the basic commands for creating and
// interacting with an ostree repository
package otbuiltin
import (
"errors"
"fmt"
"runtime"
"unsafe"
glib "github.com/ostreedev/ostree-go/pkg/glibobject"
)
// #cgo pkg-config: ostree-1
// #include <stdlib.h>
// #include <glib.h>
// #include <ostree.h>
// #include "builtin.go.h"
import "C"
// Repo represents a local ostree repository
type Repo struct {
ptr unsafe.Pointer
}
// isInitialized checks if the repo has been initialized
func (r *Repo) isInitialized() bool {
if r == nil || r.ptr == nil {
return false
}
return true
}
// native converts an ostree repo struct to its C equivalent
func (r *Repo) native() *C.OstreeRepo {
if !r.isInitialized() {
return nil
}
return (*C.OstreeRepo)(r.ptr)
}
// repoFromNative takes a C ostree repo and converts it to a Go struct
func repoFromNative(or *C.OstreeRepo) *Repo {
if or == nil {
return nil
}
r := &Repo{unsafe.Pointer(or)}
return r
}
// OpenRepo attempts to open the repo at the given path
func OpenRepo(path string) (*Repo, error) {
if path == "" {
return nil, errors.New("empty path")
}
cpath := C.CString(path)
defer C.free(unsafe.Pointer(cpath))
repoPath := C.g_file_new_for_path(cpath)
defer C.g_object_unref(C.gpointer(repoPath))
crepo := C.ostree_repo_new(repoPath)
repo := repoFromNative(crepo)
var cerr *C.GError
r := glib.GoBool(glib.GBoolean(C.ostree_repo_open(crepo, nil, &cerr)))
if !r {
return nil, generateError(cerr)
}
return repo, nil
}
// enableTombstoneCommits enables support for tombstone commits.
//
// This allows to distinguish between intentional deletions and accidental removals
// of commits.
func (r *Repo) enableTombstoneCommits() error {
if !r.isInitialized() {
return errors.New("repo not initialized")
}
config := C.ostree_repo_get_config(r.native())
groupC := C.CString("core")
defer C.free(unsafe.Pointer(groupC))
keyC := C.CString("tombstone-commits")
defer C.free(unsafe.Pointer(keyC))
valueC := C.g_key_file_get_boolean(config, (*C.gchar)(groupC), (*C.gchar)(keyC), nil)
tombstoneCommits := glib.GoBool(glib.GBoolean(valueC))
// tombstoneCommits is false only if it really is false or if it is set to FALSE in the config file
if !tombstoneCommits {
var cerr *C.GError
C.g_key_file_set_boolean(config, (*C.gchar)(groupC), (*C.gchar)(keyC), C.TRUE)
if !glib.GoBool(glib.GBoolean(C.ostree_repo_write_config(r.native(), config, &cerr))) {
return generateError(cerr)
}
}
return nil
}
// generateError wraps a GLib error into a Go one.
func generateError(err *C.GError) error {
if err == nil {
return errors.New("nil GError")
}
goErr := glib.ConvertGError(glib.ToGError(unsafe.Pointer(err)))
_, file, line, ok := runtime.Caller(1)
if ok {
return fmt.Errorf("%s:%d - %s", file, line, goErr)
}
return goErr
}
// isOk wraps a gboolean return value into a bool.
// 0 is false/error, everything else is true/ok.
func isOk(v C.gboolean) bool {
return glib.GoBool(glib.GBoolean(v))
}
|