Source code for uv.reader.base

#!/usr/bin/python
#
# Copyright 2020 Google LLC
#
# 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.
"""The base AbstractReader class, along with all of the subclasses that
AbstractReader is able to return from any of its methods."""

from abc import ABC, ABCMeta, abstractmethod
from typing import Dict, Iterable, List

import uv.types as t
import uv.util.attachment as a


[docs]class IterableReader(ABC): """Class marker for Reader instances that are able to provide an iterable of all keys that it's possible to read using this Reader instance. """
[docs] @abstractmethod def keys(self) -> Iterable[t.MetricKey]: """Returns an iterable of all keys that you're able to query from this store. """
[docs]class AbstractReader(metaclass=ABCMeta): """Base class for all classes that are able to track and return lists of metrics, keyed by a supplied t.MetricKey. NOTE - by default, read_all and read are implemented in terms of one another. This means that you can choose which method you'd like to override, or override both... but if you don't override any you'll see infinite recursion. Be careful not to abuse the kindness! """
[docs] def read_all(self, ks: List[t.MetricKey]) -> Dict[t.MetricKey, List[t.Metric]]: """Accepts a list of t.MetricKey instances and returns a mapping of t.MetricKey to a list of all metrics ever logged using that MetricKey. The interface requires that the set of keys in the returned dictionary equal the input set; any missing key in the store has to map to a default empty list. """ return {k: self.read(k) for k in ks}
[docs] def read(self, k: t.MetricKey) -> List[t.Metric]: """Returns a list of all metrics ever logged for the supplied t.MetricKey, or an empty list if the backing store has no record. """ return self.read_all([k])[k]
[docs] def close(self) -> None: """Release any resources held open by this reader instance.""" return None
# Combinators
[docs] def with_prefix(self, prefix: t.Prefix): """"Returns a new reader that will pass through all methods to this reader instance; the only difference will be that every t.MetricKey instance supplied to read or read_all will have the supplied prefix attached. """ return PrefixedReader(self, prefix)
[docs] def with_suffix(self, suffix: t.Suffix): """"Returns a new reader that will pass through all methods to this reader instance; the only difference will be that every t.MetricKey instance supplied to read or read_all will have the supplied suffix attached. """ return SuffixedReader(self, suffix)
[docs]class PrefixedReader(AbstractReader): """Reader that prepends a prefix to all keys before passing requests on to the supplied backing store. Args: base: Backing reader. All read and read_all calls proxy here. prefix: the prefix to attach to all keys supplied to any method. """ def __init__(self, base: AbstractReader, prefix: t.Prefix): self._base = base self._prefix = prefix
[docs] def read_all(self, ks: List[t.MetricKey]) -> Dict[t.MetricKey, List[t.Metric]]: return self._base.read_all(a.attach_iter(ks, self._prefix, prefix=True))
[docs] def read(self, k: t.MetricKey) -> List[t.Metric]: return self._base.read(a.attach_s(k, self._prefix, prefix=True))
[docs] def close(self) -> None: self._base.close()
[docs]class SuffixedReader(AbstractReader): """Reader that prepends a suffix to all keys before passing requests on to the supplied backing store. Args: base: Backing reader. All read and read_all calls proxy here. suffix: the suffix to attach to all keys supplied to any method. """ def __init__(self, base: AbstractReader, suffix: t.Suffix): self._base = base self._suffix = suffix
[docs] def read_all(self, ks: List[t.MetricKey]) -> Dict[t.MetricKey, List[t.Metric]]: return self._base.read_all(a.attach_iter(ks, self._suffix, prefix=False))
[docs] def read(self, k: t.MetricKey) -> List[t.Metric]: return self._base.read(a.attach_s(k, self._suffix, prefix=False))
[docs] def close(self) -> None: self._base.close()