blob: f8f2efc91e22f36e11774b082d30446b058f7fd1 [file] [log] [blame]
# ==================================================================================
# Copyright (c) 2020 Nokia
# Copyright (c) 2020 AT&T Intellectual Property.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
# ==================================================================================
"""
sdl functionality
"""
import msgpack
from ricsdl.syncstorage import SyncStorage
class SDLWrapper:
"""
This is a wrapper around the SDL Python interface.
We do not embed the below directly in the Xapp classes because
this SDL wrapper is useful for other python apps, for example A1
Mediator uses this verbatim. Therefore, we leave this here as a
seperate instantiable object so it can be used outside of xapps
too. One could argue this get moved into *sdl itself*.
We currently use msgpack for binary (de)serialization:
https://msgpack.org/index.html
"""
def __init__(self, use_fake_sdl=False):
"""
init
Parameters
----------
use_fake_sdl: bool
if this is True (default: False), then SDLs "fake dict
backend" is used, which is very useful for testing since
it allows you to use SDL without any SDL or Redis deployed at
all. This can be used while developing your xapp, and also
for monkeypatching during unit testing (e.g., the xapp
framework unit tests do this).
"""
if use_fake_sdl:
self._sdl = SyncStorage(fake_db_backend="dict")
else:
self._sdl = SyncStorage()
def set(self, ns, key, value, usemsgpack=True):
"""
sets a key
TODO: discuss whether usemsgpack should *default* to True or
False here. This seems like a usage statistic question (that we
don't have enough data for yet). Are more uses for an xapp to
write/read their own data, or will more xapps end up reading data
written by some other thing? I think it's too early to know
this. So we go with True as the very first user of this, a1, does
this. I'm open to changing this default to False later with
evidence.
Parameters
----------
ns: string
the sdl namespace
key: string
the sdl key
value:
if usemsgpack is True, value can be anything serializable by msgpack
if usemsgpack is False, value must be bytes
usemsgpack: boolean (optional)
determines whether the value is serialized using msgpack
"""
if usemsgpack:
self._sdl.set(ns, {key: msgpack.packb(value, use_bin_type=True)})
else:
self._sdl.set(ns, {key: value})
def get(self, ns, key, usemsgpack=True):
"""
get a key
Parameters
----------
ns: string
the sdl namespace
key: string
the sdl key
usemsgpack: boolean (optional)
if usemsgpack is True, the value is deserialized using msgpack
if usemsgpack is False, the value is returned as raw bytes
Returns
-------
None (if not exist) or see above; depends on usemsgpack
"""
ret_dict = self._sdl.get(ns, {key})
if key in ret_dict:
if usemsgpack:
return msgpack.unpackb(ret_dict[key], raw=False)
return ret_dict[key]
return None
def find_and_get(self, ns, prefix, usemsgpack=True):
"""
get all k v pairs that start with prefix
Parameters
----------
ns: string
the sdl namespace
key: string
the sdl key
prefix: string
the prefix
usemsgpack: boolean (optional)
if usemsgpack is True, the value returned is a dict where each value has been deserialized using msgpack
if usemsgpack is False, the value returned is as a dict mapping keys to raw bytes
Returns
-------
{} (if no keys match) or see above; depends on usemsgpack
"""
# note: SDL "*" usage is inconsistent with real python regex, where it would be ".*"
ret_dict = self._sdl.find_and_get(ns, "{0}*".format(prefix))
if usemsgpack:
return {k: msgpack.unpackb(v, raw=False) for k, v in ret_dict.items()}
return ret_dict
def delete(self, ns, key):
"""
delete a key
Parameters
----------
ns: string
the sdl namespace
key: string
the sdl key
"""
self._sdl.remove(ns, {key})
def healthcheck(self):
"""
checks if the sdl connection is healthy
Returns
-------
bool
"""
return self._sdl.is_active()