From 4cee38534919e34f407363ac3ab5f31b4d09be6d Mon Sep 17 00:00:00 2001 From: Renuka Manavalan <47282725+renukamanavalan@users.noreply.github.com> Date: Thu, 7 Mar 2019 15:04:07 -0800 Subject: [PATCH] Enhance SDK to read from any DB by name (#49) --- src/swsssdk/config/database.json | 25 +++++++++----- src/swsssdk/configdb.py | 57 ++++++++++++++++++++++++-------- 2 files changed, 61 insertions(+), 21 deletions(-) diff --git a/src/swsssdk/config/database.json b/src/swsssdk/config/database.json index 2839aad17905..1051192ff172 100644 --- a/src/swsssdk/config/database.json +++ b/src/swsssdk/config/database.json @@ -2,32 +2,41 @@ "SonicV1Connector": { "db_map": { "IF_COUNTER_DB": { - "db": 0 + "db": 0, + "separator": ":" }, "LLDP_COUNTER_DB": { - "db": 1 + "db": 1, + "separator": ":" } } }, "SonicV2Connector": { "db_map": { "APPL_DB": { - "db": 0 + "db": 0, + "separator": ":" }, "ASIC_DB": { - "db": 1 + "db": 1, + "separator": ":" }, "COUNTERS_DB": { - "db": 2 + "db": 2, + "separator": ":" }, "CONFIG_DB": { - "db": 4 + "db": 4, + "separator": "|" }, "STATE_DB": { - "db": 6 + "db": 6, + "separator": "|" }, "SNMP_OVERLAY_DB": { - "db": 7 + "db": 7, + "separator": "|" + } } } diff --git a/src/swsssdk/configdb.py b/src/swsssdk/configdb.py index f2626f2d1549..2fa896f25ab5 100644 --- a/src/swsssdk/configdb.py +++ b/src/swsssdk/configdb.py @@ -37,11 +37,11 @@ def __init__(self, **kwargs): self.handlers = {} def __wait_for_db_init(self): - client = self.redis_clients[self.CONFIG_DB] + client = self.redis_clients[self.db_name] pubsub = client.pubsub() initialized = client.get(self.INIT_INDICATOR) if not initialized: - pattern = "__keyspace@{}__:{}".format(self.db_map[self.CONFIG_DB]['db'], self.INIT_INDICATOR) + pattern = "__keyspace@{}__:{}".format(self.db_map[self.db_name]['db'], self.INIT_INDICATOR) pubsub.psubscribe(pattern) for item in pubsub.listen(): if item['type'] == 'pmessage': @@ -53,11 +53,16 @@ def __wait_for_db_init(self): pubsub.punsubscribe(pattern) - def connect(self, wait_for_init=True, retry_on=False): - SonicV2Connector.connect(self, self.CONFIG_DB, retry_on) + def db_connect(self, dbname, wait_for_init=False, retry_on=False): + self.db_name = dbname + self.KEY_SEPARATOR = self.TABLE_NAME_SEPARATOR = self.db_map[self.db_name]['separator'] + SonicV2Connector.connect(self, self.db_name, retry_on) if wait_for_init: self.__wait_for_db_init() + def connect(self, wait_for_init=True, retry_on=False): + self.db_connect('CONFIG_DB', wait_for_init, retry_on) + def subscribe(self, table, handler): """Set a handler to handle config change in certain table. Note that a single handler can be registered to different tables by @@ -84,15 +89,15 @@ def __fire(self, table, key, data): def listen(self): """Start listen Redis keyspace events and will trigger corresponding handlers when content of a table changes. """ - self.pubsub = self.redis_clients[self.CONFIG_DB].pubsub() - self.pubsub.psubscribe("__keyspace@{}__:*".format(self.db_map[self.CONFIG_DB]['db'])) + self.pubsub = self.redis_clients[self.db_name].pubsub() + self.pubsub.psubscribe("__keyspace@{}__:*".format(self.db_map[self.db_name]['db'])) for item in self.pubsub.listen(): if item['type'] == 'pmessage': key = item['channel'].split(':', 1)[1] try: (table, row) = key.split(self.TABLE_NAME_SEPARATOR, 1) if self.handlers.has_key(table): - client = self.redis_clients[self.CONFIG_DB] + client = self.redis_clients[self.db_name] data = self.__raw_to_typed(client.hgetall(key)) self.__fire(table, row, data) except ValueError: @@ -166,7 +171,7 @@ def set_entry(self, table, key, data): Pass None as data will delete the entry. """ key = self.serialize_key(key) - client = self.redis_clients[self.CONFIG_DB] + client = self.redis_clients[self.db_name] _hash = '{}{}{}'.format(table.upper(), self.TABLE_NAME_SEPARATOR, key) if data == None: client.delete(_hash) @@ -188,7 +193,7 @@ def mod_entry(self, table, key, data): Pass None as data will delete the entry. """ key = self.serialize_key(key) - client = self.redis_clients[self.CONFIG_DB] + client = self.redis_clients[self.db_name] _hash = '{}{}{}'.format(table.upper(), self.TABLE_NAME_SEPARATOR, key) if data == None: client.delete(_hash) @@ -205,10 +210,36 @@ def get_entry(self, table, key): Empty dictionary if table does not exist or entry does not exist. """ key = self.serialize_key(key) - client = self.redis_clients[self.CONFIG_DB] + client = self.redis_clients[self.db_name] _hash = '{}{}{}'.format(table.upper(), self.TABLE_NAME_SEPARATOR, key) return self.__raw_to_typed(client.hgetall(_hash)) + def get_keys(self, table, split=True): + """Read all keys of a table from config db. + Args: + table: Table name. + split: split the first part and return second. + Useful for keys with two parts : + Returns: + List of keys. + """ + client = self.redis_clients[self.db_name] + pattern = '{}{}*'.format(table.upper(), self.TABLE_NAME_SEPARATOR) + keys = client.keys(pattern) + data = [] + for key in keys: + try: + if PY3K: + key = key.decode('utf-8') + if split: + (_, row) = key.split(self.TABLE_NAME_SEPARATOR, 1) + data.append(self.deserialize_key(row)) + else: + data.append(self.deserialize_key(key)) + except ValueError: + pass #Ignore non table-formated redis entries + return data + def get_table(self, table): """Read an entire table from config db. Args: @@ -219,7 +250,7 @@ def get_table(self, table): or { ('l1_key', 'l2_key', ...): {'column_key': value, ...}, ...} for a multi-key table. Empty dictionary if table does not exist. """ - client = self.redis_clients[self.CONFIG_DB] + client = self.redis_clients[self.db_name] pattern = '{}{}*'.format(table.upper(), self.TABLE_NAME_SEPARATOR) keys = client.keys(pattern) data = {} @@ -243,7 +274,7 @@ def delete_table(self, table): Args: table: Table name. """ - client = self.redis_clients[self.CONFIG_DB] + client = self.redis_clients[self.db_name] pattern = '{}{}*'.format(table.upper(), self.TABLE_NAME_SEPARATOR) keys = client.keys(pattern) data = {} @@ -276,7 +307,7 @@ def get_config(self): ... } """ - client = self.redis_clients[self.CONFIG_DB] + client = self.redis_clients[self.db_name] keys = client.keys('*') data = {} for key in keys: