""" sdl functionality """ # ================================================================================== # 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. # ================================================================================== 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): """ set a key NOTE: I am down for a discussion about 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()