/
store.go
135 lines (114 loc) · 3.12 KB
/
store.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
// Copyright (C) 2015-2020 the Gprovision Authors. All Rights Reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
//
// SPDX-License-Identifier: BSD-3-Clause
//
package server
import (
"strings"
"sync"
"github.com/purecloudlabs/gprovision/pkg/log"
"github.com/purecloudlabs/gprovision/pkg/oss/pblog/pb"
"github.com/golang/protobuf/proto"
"github.com/prologic/bitcask"
)
// A trivial database impl for our data storage needs. Not recommended for any
// serious use. Sensitive information stored as PLAINTEXT. No auth. Etc etc etc.
//
// All entries stored as protobuf; all items in one db.
// Log entries stored read-append-write, likely to perform very poorly at scale.
type Dbase struct {
bc *bitcask.Bitcask
sync.Mutex
}
func OpenDB(path string) *Dbase {
db, err := bitcask.Open(path)
if err != nil {
panic(err)
}
return &Dbase{bc: db}
}
func (db *Dbase) StoreLog(id string, les *pb.LogEvents) error {
return db.append(key(id, "log"), les)
}
func (db *Dbase) RetrieveLog(id string) (m pb.LogEvents, err error) {
err = db.deserialize(key(id, "log"), &m)
return
}
func (db *Dbase) StoreMacs(id string, m pb.MACs) error {
return db.serialize(key(id, "mac"), &m)
}
func (db *Dbase) RetrieveMacs(id string) (m pb.MACs, err error) {
err = db.deserialize(key(id, "mac"), &m)
return
}
func (db *Dbase) StoreIpmiMacs(id string, m pb.MACs) error {
return db.serialize(key(id, "imac"), &m)
}
func (db *Dbase) RetrieveIpmiMacs(id string) (m pb.MACs, err error) {
err = db.deserialize(key(id, "imac"), &m)
return
}
const credwarn = "WARNING storage of credentials as plain text is a bad idea."
//WARNING storage of credentials as plain text is a bad idea.
func (db *Dbase) StorePass(id string, p *pb.Credentials) error {
log.Log(credwarn)
return db.serialize(key(id, "pas"), p)
}
//WARNING storage of credentials as plain text is a bad idea.
func (db *Dbase) RetrievePass(id string) (p *pb.Credentials, err error) {
log.Log(credwarn)
err = db.deserialize(key(id, "pas"), p)
return
}
//return all ids that have been used when logging or reporting macs/ipmi macs
func (db *Dbase) Ids() []string {
//use a map to deduplicate
ids := make(map[string]interface{})
for k := range db.bc.Keys() {
sp := strings.Split(string(k), "_")
ids[sp[0]] = nil
}
var idlist []string
for k := range ids {
idlist = append(idlist, k)
}
return idlist
}
func (db *Dbase) Close() error {
db.Lock()
defer db.Unlock()
return db.bc.Close()
}
func (db *Dbase) serialize(k []byte, m proto.Message) error {
bytes, err := proto.Marshal(m)
if err != nil {
return err
}
db.Lock()
defer db.Unlock()
return db.bc.Put(k, bytes)
}
func (db *Dbase) deserialize(k []byte, m proto.Message) error {
db.Lock()
v, err := db.bc.Get(k)
db.Unlock()
if err != nil {
return err
}
err = proto.Unmarshal(v, m)
return err
}
func (db *Dbase) append(k []byte, msg proto.Message) error {
db.Lock()
defer db.Unlock()
v, _ := db.bc.Get(k)
m, err := proto.Marshal(msg)
if err != nil {
return err
}
v = append(v, m...)
return db.bc.Put(k, v)
}
func key(id, typ string) []byte { return []byte(id + "_" + typ) }