| 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
 | package pgdialect
import (
	"database/sql"
	"fmt"
	"reflect"
	"github.com/uptrace/bun/schema"
)
type HStoreValue struct {
	v reflect.Value
	append schema.AppenderFunc
	scan   schema.ScannerFunc
}
// HStore accepts a map[string]string and returns a wrapper for working with PostgreSQL
// hstore data type.
//
// For struct fields you can use hstore tag:
//
//    Attrs  map[string]string `bun:",hstore"`
func HStore(vi interface{}) *HStoreValue {
	v := reflect.ValueOf(vi)
	if !v.IsValid() {
		panic(fmt.Errorf("bun: HStore(nil)"))
	}
	typ := v.Type()
	if typ.Kind() == reflect.Ptr {
		typ = typ.Elem()
	}
	if typ.Kind() != reflect.Map {
		panic(fmt.Errorf("bun: Hstore(unsupported %s)", typ))
	}
	return &HStoreValue{
		v: v,
		append: pgDialect.hstoreAppender(v.Type()),
		scan:   hstoreScanner(v.Type()),
	}
}
var (
	_ schema.QueryAppender = (*HStoreValue)(nil)
	_ sql.Scanner          = (*HStoreValue)(nil)
)
func (h *HStoreValue) AppendQuery(fmter schema.Formatter, b []byte) ([]byte, error) {
	if h.append == nil {
		panic(fmt.Errorf("bun: HStore(unsupported %s)", h.v.Type()))
	}
	return h.append(fmter, b, h.v), nil
}
func (h *HStoreValue) Scan(src interface{}) error {
	if h.scan == nil {
		return fmt.Errorf("bun: HStore(unsupported %s)", h.v.Type())
	}
	if h.v.Kind() != reflect.Ptr {
		return fmt.Errorf("bun: HStore(non-pointer %s)", h.v.Type())
	}
	return h.scan(h.v.Elem(), src)
}
func (h *HStoreValue) Value() interface{} {
	if h.v.IsValid() {
		return h.v.Interface()
	}
	return nil
}
 |