-
Notifications
You must be signed in to change notification settings - Fork 7
/
config.go
165 lines (138 loc) · 4.24 KB
/
config.go
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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
package tiledb
/*
#include <tiledb/tiledb.h>
#include <stdlib.h>
*/
import "C"
import (
"fmt"
"unsafe"
)
// Config carries configuration parameters for a context.
type Config struct {
tiledbConfig *C.tiledb_config_t
}
// NewConfig allocates a new configuration.
func NewConfig() (*Config, error) {
var config Config
var err *C.tiledb_error_t
C.tiledb_config_alloc(&config.tiledbConfig, &err)
if err != nil {
var msg *C.char
C.tiledb_error_message(err, &msg)
defer C.tiledb_error_free(&err)
return nil, fmt.Errorf("error creating tiledb config: %s", C.GoString(msg))
}
freeOnGC(&config)
return &config, nil
}
// Set sets a config parameter-value pair.
func (c *Config) Set(param string, value string) error {
var err *C.tiledb_error_t
cparam := C.CString(param)
defer C.free(unsafe.Pointer(cparam))
cvalue := C.CString(value)
defer C.free(unsafe.Pointer(cvalue))
C.tiledb_config_set(c.tiledbConfig, cparam, cvalue, &err)
if err != nil {
var msg *C.char
C.tiledb_error_message(err, &msg)
defer C.tiledb_error_free(&err)
return fmt.Errorf("error setting %s:%s in config: %s", param, value, C.GoString(msg))
}
return nil
}
// Get gets a parameter from the configuration by key.
func (c *Config) Get(param string) (string, error) {
var err *C.tiledb_error_t
var val *C.char
cparam := C.CString(param)
defer C.free(unsafe.Pointer(cparam))
C.tiledb_config_get(c.tiledbConfig, cparam, &val, &err)
if err != nil {
var msg *C.char
C.tiledb_error_message(err, &msg)
defer C.tiledb_error_free(&err)
return "", fmt.Errorf("error getting %s in config: %s", param, C.GoString(msg))
}
value := C.GoString(val)
return value, nil
}
// Unset resets a config parameter to its default value.
func (c *Config) Unset(param string) error {
var err *C.tiledb_error_t
cparam := C.CString(param)
defer C.free(unsafe.Pointer(cparam))
C.tiledb_config_unset(c.tiledbConfig, cparam, &err)
if err != nil {
var msg *C.char
C.tiledb_error_message(err, &msg)
defer C.tiledb_error_free(&err)
return fmt.Errorf("error unsetting %s in config: %s", param, C.GoString(msg))
}
return nil
}
// SaveToFile saves the config parameters to a (local) text file.
func (c *Config) SaveToFile(file string) error {
var err *C.tiledb_error_t
cfile := C.CString(file)
defer C.free(unsafe.Pointer(cfile))
C.tiledb_config_save_to_file(c.tiledbConfig, cfile, &err)
if err != nil {
var msg *C.char
C.tiledb_error_message(err, &msg)
defer C.tiledb_error_free(&err)
return fmt.Errorf("error saving config from file %s: %s", file, C.GoString(msg))
}
return nil
}
// LoadConfig reads a configuration from the given uri.
func LoadConfig(uri string) (*Config, error) {
if uri == "" {
return nil, fmt.Errorf("error loading tiledb config: passed uri is empty")
}
var config Config
var err *C.tiledb_error_t
C.tiledb_config_alloc(&config.tiledbConfig, &err)
if err != nil {
var msg *C.char
C.tiledb_error_message(err, &msg)
defer C.tiledb_error_free(&err)
return nil, fmt.Errorf("error loading tiledb config: %s", C.GoString(msg))
}
curi := C.CString(uri)
defer C.free(unsafe.Pointer(curi))
C.tiledb_config_load_from_file(config.tiledbConfig, curi, &err)
if err != nil {
var msg *C.char
C.tiledb_error_message(err, &msg)
defer C.tiledb_error_free(&err)
return nil, fmt.Errorf("error loading config from file %s: %s", uri, C.GoString(msg))
}
freeOnGC(&config)
return &config, nil
}
// Free releases the internal TileDB core data that was allocated on the C heap.
// It is automatically called when this object is garbage collected, but can be
// called earlier to manually release memory if needed. Free is idempotent and
// can safely be called many times on the same object; if it has already
// been freed, it will not be freed again.
func (c *Config) Free() {
if c.tiledbConfig != nil {
C.tiledb_config_free(&c.tiledbConfig)
}
}
// Iterate iterates over configuration.
//
// for iter, err := config.Iterate(); !iter.Done(); iter.Next(){
// param, value, err := iter.Here()
// }
func (c *Config) Iterate(prefix string) (*ConfigIter, error) {
return NewConfigIter(c, prefix)
}
// Cmp compares two configs.
func (c *Config) Cmp(other *Config) bool {
var equal C.uint8_t
C.tiledb_config_compare(c.tiledbConfig, other.tiledbConfig, &equal)
return equal == 1
}