blob: dc24fe65ddd349da1bd8860e2c41beb7ebfba0f7 [file] [log] [blame]
divyadesai81bb7ba2020-03-11 11:45:23 +00001/*
Joey Armstrong43f5cc02024-01-12 18:33:25 -05002 * Copyright 2018-2024 Open Networking Foundation (ONF) and the ONF Contributors
divyadesai81bb7ba2020-03-11 11:45:23 +00003
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7
8 * http://www.apache.org/licenses/LICENSE-2.0
9
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package db
18
19import (
20 "context"
21 "errors"
22 "fmt"
Girish Kumarcd402012020-08-18 12:17:38 +000023 "sync"
divyadesai81bb7ba2020-03-11 11:45:23 +000024 "time"
25
David K. Bainbridgee05cf0c2021-08-19 03:16:50 +000026 "github.com/opencord/voltha-lib-go/v7/pkg/db/kvstore"
27 "github.com/opencord/voltha-lib-go/v7/pkg/log"
divyadesai81bb7ba2020-03-11 11:45:23 +000028 "google.golang.org/grpc/codes"
29 "google.golang.org/grpc/status"
30)
31
32const (
33 // Default Minimal Interval for posting alive state of backend kvstore on Liveness Channel
34 DefaultLivenessChannelInterval = time.Second * 30
35)
36
37// Backend structure holds details for accessing the kv store
38type Backend struct {
divyadesai81bb7ba2020-03-11 11:45:23 +000039 Client kvstore.Client
40 StoreType string
Neha Sharma87d43d72020-04-08 14:10:40 +000041 Timeout time.Duration
Neha Sharma318a1292020-05-14 19:53:26 +000042 Address string
divyadesai81bb7ba2020-03-11 11:45:23 +000043 PathPrefix string
Girish Kumarcd402012020-08-18 12:17:38 +000044 alive bool // Is this backend connection alive?
45 livenessMutex sync.Mutex
divyadesai81bb7ba2020-03-11 11:45:23 +000046 liveness chan bool // channel to post alive state
47 LivenessChannelInterval time.Duration // regularly push alive state beyond this interval
48 lastLivenessTime time.Time // Instant of last alive state push
49}
50
51// NewBackend creates a new instance of a Backend structure
Rohan Agrawalc32d9932020-06-15 11:01:47 +000052func NewBackend(ctx context.Context, storeType string, address string, timeout time.Duration, pathPrefix string) *Backend {
divyadesai81bb7ba2020-03-11 11:45:23 +000053 var err error
54
55 b := &Backend{
56 StoreType: storeType,
Neha Sharma318a1292020-05-14 19:53:26 +000057 Address: address,
divyadesai81bb7ba2020-03-11 11:45:23 +000058 Timeout: timeout,
59 LivenessChannelInterval: DefaultLivenessChannelInterval,
60 PathPrefix: pathPrefix,
61 alive: false, // connection considered down at start
62 }
63
Rohan Agrawalc32d9932020-06-15 11:01:47 +000064 if b.Client, err = b.newClient(ctx, address, timeout); err != nil {
65 logger.Errorw(ctx, "failed-to-create-kv-client",
divyadesai81bb7ba2020-03-11 11:45:23 +000066 log.Fields{
Neha Sharma318a1292020-05-14 19:53:26 +000067 "type": storeType, "address": address,
divyadesai81bb7ba2020-03-11 11:45:23 +000068 "timeout": timeout, "prefix": pathPrefix,
69 "error": err.Error(),
70 })
71 }
72
73 return b
74}
75
Rohan Agrawalc32d9932020-06-15 11:01:47 +000076func (b *Backend) newClient(ctx context.Context, address string, timeout time.Duration) (kvstore.Client, error) {
divyadesai81bb7ba2020-03-11 11:45:23 +000077 switch b.StoreType {
Mahir Gunyel82f98dd2023-07-24 10:29:49 +030078 case "redis":
79 return kvstore.NewRedisClient(address, timeout, false)
80 case "redis-sentinel":
81 return kvstore.NewRedisClient(address, timeout, true)
divyadesai81bb7ba2020-03-11 11:45:23 +000082 case "etcd":
Rohan Agrawalc32d9932020-06-15 11:01:47 +000083 return kvstore.NewEtcdClient(ctx, address, timeout, log.WarnLevel)
divyadesai81bb7ba2020-03-11 11:45:23 +000084 }
85 return nil, errors.New("unsupported-kv-store")
86}
87
Rohan Agrawalc32d9932020-06-15 11:01:47 +000088func (b *Backend) makePath(ctx context.Context, key string) string {
divyadesai81bb7ba2020-03-11 11:45:23 +000089 path := fmt.Sprintf("%s/%s", b.PathPrefix, key)
90 return path
91}
92
Rohan Agrawalc32d9932020-06-15 11:01:47 +000093func (b *Backend) updateLiveness(ctx context.Context, alive bool) {
divyadesai81bb7ba2020-03-11 11:45:23 +000094 // Periodically push stream of liveness data to the channel,
95 // so that in a live state, the core does not timeout and
96 // send a forced liveness message. Push alive state if the
97 // last push to channel was beyond livenessChannelInterval
Girish Kumarcd402012020-08-18 12:17:38 +000098 b.livenessMutex.Lock()
99 defer b.livenessMutex.Unlock()
divyadesai81bb7ba2020-03-11 11:45:23 +0000100 if b.liveness != nil {
divyadesai81bb7ba2020-03-11 11:45:23 +0000101 if b.alive != alive {
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000102 logger.Debug(ctx, "update-liveness-channel-reason-change")
divyadesai81bb7ba2020-03-11 11:45:23 +0000103 b.liveness <- alive
104 b.lastLivenessTime = time.Now()
105 } else if time.Since(b.lastLivenessTime) > b.LivenessChannelInterval {
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000106 logger.Debug(ctx, "update-liveness-channel-reason-interval")
divyadesai81bb7ba2020-03-11 11:45:23 +0000107 b.liveness <- alive
108 b.lastLivenessTime = time.Now()
109 }
110 }
111
112 // Emit log message only for alive state change
113 if b.alive != alive {
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000114 logger.Debugw(ctx, "change-kvstore-alive-status", log.Fields{"alive": alive})
divyadesai81bb7ba2020-03-11 11:45:23 +0000115 b.alive = alive
116 }
117}
118
119// Perform a dummy Key Lookup on kvstore to test Connection Liveness and
120// post on Liveness channel
121func (b *Backend) PerformLivenessCheck(ctx context.Context) bool {
122 alive := b.Client.IsConnectionUp(ctx)
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000123 logger.Debugw(ctx, "kvstore-liveness-check-result", log.Fields{"alive": alive})
divyadesai81bb7ba2020-03-11 11:45:23 +0000124
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000125 b.updateLiveness(ctx, alive)
divyadesai81bb7ba2020-03-11 11:45:23 +0000126 return alive
127}
128
129// Enable the liveness monitor channel. This channel will report
130// a "true" or "false" on every kvstore operation which indicates whether
131// or not the connection is still Live. This channel is then picked up
132// by the service (i.e. rw_core / ro_core) to update readiness status
133// and/or take other actions.
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000134func (b *Backend) EnableLivenessChannel(ctx context.Context) chan bool {
135 logger.Debug(ctx, "enable-kvstore-liveness-channel")
Girish Kumarcd402012020-08-18 12:17:38 +0000136 b.livenessMutex.Lock()
137 defer b.livenessMutex.Unlock()
divyadesai81bb7ba2020-03-11 11:45:23 +0000138 if b.liveness == nil {
divyadesai81bb7ba2020-03-11 11:45:23 +0000139 b.liveness = make(chan bool, 10)
divyadesai81bb7ba2020-03-11 11:45:23 +0000140 b.liveness <- b.alive
141 b.lastLivenessTime = time.Now()
142 }
143
144 return b.liveness
145}
146
147// Extract Alive status of Kvstore based on type of error
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000148func (b *Backend) isErrorIndicatingAliveKvstore(ctx context.Context, err error) bool {
divyadesai81bb7ba2020-03-11 11:45:23 +0000149 // Alive unless observed an error indicating so
150 alive := true
151
152 if err != nil {
153
154 // timeout indicates kvstore not reachable/alive
155 if err == context.DeadlineExceeded {
156 alive = false
157 }
158
159 // Need to analyze client-specific errors based on backend type
160 if b.StoreType == "etcd" {
161
162 // For etcd backend, consider not-alive only for errors indicating
163 // timedout request or unavailable/corrupted cluster. For all remaining
164 // error codes listed in https://godoc.org/google.golang.org/grpc/codes#Code,
165 // we would not infer a not-alive backend because such a error may also
166 // occur due to bad client requests or sequence of operations
167 switch status.Code(err) {
168 case codes.DeadlineExceeded:
169 fallthrough
170 case codes.Unavailable:
171 fallthrough
172 case codes.DataLoss:
173 alive = false
174 }
divyadesai81bb7ba2020-03-11 11:45:23 +0000175 }
176 }
177
178 return alive
179}
180
181// List retrieves one or more items that match the specified key
182func (b *Backend) List(ctx context.Context, key string) (map[string]*kvstore.KVPair, error) {
Mahir Gunyel82f98dd2023-07-24 10:29:49 +0300183 span, ctx := log.CreateChildSpan(ctx, "kvs-list")
Girish Kumarcd402012020-08-18 12:17:38 +0000184 defer span.Finish()
185
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000186 formattedPath := b.makePath(ctx, key)
187 logger.Debugw(ctx, "listing-key", log.Fields{"key": key, "path": formattedPath})
divyadesai81bb7ba2020-03-11 11:45:23 +0000188
189 pair, err := b.Client.List(ctx, formattedPath)
190
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000191 b.updateLiveness(ctx, b.isErrorIndicatingAliveKvstore(ctx, err))
divyadesai81bb7ba2020-03-11 11:45:23 +0000192
193 return pair, err
194}
195
196// Get retrieves an item that matches the specified key
197func (b *Backend) Get(ctx context.Context, key string) (*kvstore.KVPair, error) {
Mahir Gunyel82f98dd2023-07-24 10:29:49 +0300198 span, ctx := log.CreateChildSpan(ctx, "kvs-get")
Girish Kumarcd402012020-08-18 12:17:38 +0000199 defer span.Finish()
200
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000201 formattedPath := b.makePath(ctx, key)
202 logger.Debugw(ctx, "getting-key", log.Fields{"key": key, "path": formattedPath})
divyadesai81bb7ba2020-03-11 11:45:23 +0000203
204 pair, err := b.Client.Get(ctx, formattedPath)
205
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000206 b.updateLiveness(ctx, b.isErrorIndicatingAliveKvstore(ctx, err))
divyadesai81bb7ba2020-03-11 11:45:23 +0000207
208 return pair, err
209}
210
211// Put stores an item value under the specifed key
212func (b *Backend) Put(ctx context.Context, key string, value interface{}) error {
Mahir Gunyel82f98dd2023-07-24 10:29:49 +0300213 span, ctx := log.CreateChildSpan(ctx, "kvs-put")
Girish Kumarcd402012020-08-18 12:17:38 +0000214 defer span.Finish()
215
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000216 formattedPath := b.makePath(ctx, key)
217 logger.Debugw(ctx, "putting-key", log.Fields{"key": key, "path": formattedPath})
divyadesai81bb7ba2020-03-11 11:45:23 +0000218
219 err := b.Client.Put(ctx, formattedPath, value)
220
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000221 b.updateLiveness(ctx, b.isErrorIndicatingAliveKvstore(ctx, err))
divyadesai81bb7ba2020-03-11 11:45:23 +0000222
223 return err
224}
225
226// Delete removes an item under the specified key
227func (b *Backend) Delete(ctx context.Context, key string) error {
Mahir Gunyel82f98dd2023-07-24 10:29:49 +0300228 span, ctx := log.CreateChildSpan(ctx, "kvs-delete")
Girish Kumarcd402012020-08-18 12:17:38 +0000229 defer span.Finish()
230
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000231 formattedPath := b.makePath(ctx, key)
232 logger.Debugw(ctx, "deleting-key", log.Fields{"key": key, "path": formattedPath})
divyadesai81bb7ba2020-03-11 11:45:23 +0000233
234 err := b.Client.Delete(ctx, formattedPath)
235
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000236 b.updateLiveness(ctx, b.isErrorIndicatingAliveKvstore(ctx, err))
divyadesai81bb7ba2020-03-11 11:45:23 +0000237
238 return err
239}
240
David K. Bainbridge595b6702021-04-09 16:10:58 +0000241func (b *Backend) DeleteWithPrefix(ctx context.Context, prefixKey string) error {
Mahir Gunyel82f98dd2023-07-24 10:29:49 +0300242 span, ctx := log.CreateChildSpan(ctx, "kvs-delete-with-prefix")
David K. Bainbridge595b6702021-04-09 16:10:58 +0000243 defer span.Finish()
244
245 formattedPath := b.makePath(ctx, prefixKey)
246 logger.Debugw(ctx, "deleting-prefix-key", log.Fields{"key": prefixKey, "path": formattedPath})
247
248 err := b.Client.DeleteWithPrefix(ctx, formattedPath)
249
250 b.updateLiveness(ctx, b.isErrorIndicatingAliveKvstore(ctx, err))
251
252 return err
253}
254
divyadesai81bb7ba2020-03-11 11:45:23 +0000255// CreateWatch starts watching events for the specified key
256func (b *Backend) CreateWatch(ctx context.Context, key string, withPrefix bool) chan *kvstore.Event {
Mahir Gunyel82f98dd2023-07-24 10:29:49 +0300257 span, ctx := log.CreateChildSpan(ctx, "kvs-create-watch")
Girish Kumarcd402012020-08-18 12:17:38 +0000258 defer span.Finish()
259
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000260 formattedPath := b.makePath(ctx, key)
261 logger.Debugw(ctx, "creating-key-watch", log.Fields{"key": key, "path": formattedPath})
divyadesai81bb7ba2020-03-11 11:45:23 +0000262
263 return b.Client.Watch(ctx, formattedPath, withPrefix)
264}
265
266// DeleteWatch stops watching events for the specified key
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000267func (b *Backend) DeleteWatch(ctx context.Context, key string, ch chan *kvstore.Event) {
Mahir Gunyel82f98dd2023-07-24 10:29:49 +0300268 span, ctx := log.CreateChildSpan(ctx, "kvs-delete-watch")
Girish Kumarcd402012020-08-18 12:17:38 +0000269 defer span.Finish()
270
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000271 formattedPath := b.makePath(ctx, key)
272 logger.Debugw(ctx, "deleting-key-watch", log.Fields{"key": key, "path": formattedPath})
divyadesai81bb7ba2020-03-11 11:45:23 +0000273
Rohan Agrawalc32d9932020-06-15 11:01:47 +0000274 b.Client.CloseWatch(ctx, formattedPath, ch)
divyadesai81bb7ba2020-03-11 11:45:23 +0000275}