-
Notifications
You must be signed in to change notification settings - Fork 15
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Introduce accumulating statistic in time-series DB
Idea is to start gathering statistic about amount of entities existing in every account at every point of time. This will allow detect different kind of anomalies in testing pipelines (e.g. bugs in scheduling , bugs in cleanup etc.) This PR just introduce functionality which allow to achieve this in future PRs we will start using it in EVERY query for EVERY cloud provider.
- Loading branch information
1 parent
eca9f33
commit 5e0eeef
Showing
7 changed files
with
89 additions
and
15 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -11,9 +11,6 @@ static/ | |
venv/ | ||
.venv/ | ||
pyvenv.cfg | ||
lib64/ | ||
lib/ | ||
bin/ | ||
|
||
# Codecov | ||
.coverage | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
import os | ||
import logging | ||
from influxdb_client import InfluxDBClient, Point | ||
from influxdb_client.client.write_api import SYNCHRONOUS, WriteApi | ||
from influxdb_client.client.exceptions import InfluxDBError | ||
from urllib3.exceptions import HTTPError, TimeoutError | ||
|
||
|
||
from webui.PCWConfig import PCWConfig | ||
from ocw.enums import ProviderChoice | ||
|
||
logger = logging.getLogger(__name__) | ||
|
||
|
||
def influxwrite(influx_type: str, provider: ProviderChoice): | ||
def decorator(func): | ||
def wrapper(*args, **kwargs): | ||
objects = func(*args, **kwargs) | ||
Influx().write(provider.value, influx_type, len(objects)) | ||
return objects | ||
return wrapper | ||
return decorator | ||
|
||
|
||
class Influx: | ||
__client: WriteApi | None = None | ||
VMS_QUANTITY: str = "vms_quantity" | ||
IMAGES_QUANTITY: str = "images_quantity" | ||
DISK_QUANTITY: str = "disk_quantity" | ||
|
||
def __init__(self) -> None: | ||
if self.__client is None: | ||
if os.getenv("INFLUX_TOKEN") is None: | ||
logger.warning("INFLUX_TOKEN is not set, InfluxDB will not be used") | ||
elif PCWConfig.has("influxdb/url"): | ||
self.bucket: str = str(PCWConfig.get_feature_property("influxdb", "bucket")) | ||
self.org: str = str(PCWConfig.get_feature_property("influxdb", "org")) | ||
url: str = str(PCWConfig.get_feature_property("influxdb", "url")) | ||
self.__client = InfluxDBClient( | ||
url=url, | ||
token=os.getenv("INFLUX_TOKEN"), | ||
org=str(PCWConfig.get_feature_property("influxdb", "org")), | ||
).write_api(write_options=SYNCHRONOUS) | ||
|
||
# this is implementation of Singleton pattern | ||
def __new__(cls: type["Influx"]) -> "Influx": | ||
if not hasattr(cls, "instance") or cls.instance is None: | ||
cls.instance = super(Influx, cls).__new__(cls) | ||
return cls.instance | ||
|
||
def write(self, measurement: str, field: str, value: int) -> None: | ||
if self.__client: | ||
point = Point(measurement).field(field, value) | ||
try: | ||
self.__client.write(bucket=self.bucket, org=self.org, record=point) | ||
except (InfluxDBError, HTTPError, TimeoutError) as exception: | ||
logger.warning(f"Failed to write to influxdb(record={point}): {exception}") |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -21,3 +21,4 @@ openstacksdk~=1.5.0 | |
python-dateutil | ||
apscheduler | ||
kubernetes | ||
influxdb-client |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
from ocw.lib.influx import Influx | ||
|
||
|
||
def test_influx_init(): | ||
influx = Influx() | ||
assert hasattr(influx, "__client") is False |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -53,8 +53,8 @@ def getList(self, config_path: str, default: list = None) -> list: | |
class PCWConfig(): | ||
|
||
@staticmethod | ||
def get_feature_property(feature: str, property: str, namespace: str = None): | ||
default_values = { | ||
def get_feature_property(feature: str, feature_property: str, namespace: str | None = None) -> str | int: | ||
default_values: dict[str, dict[str, int | type[int] | str | type[str] | type[str] | None]] = { | ||
'cleanup/max-age-hours': {'default': 24 * 7, 'return_type': int}, | ||
'cleanup/azure-gallery-name': {'default': 'test_image_gallery', 'return_type': str}, | ||
'cleanup/azure-storage-resourcegroup': {'default': 'openqa-upload', 'return_type': str}, | ||
|
@@ -70,12 +70,15 @@ def get_feature_property(feature: str, property: str, namespace: str = None): | |
'notify/smtp': {'default': None, 'return_type': str}, | ||
'notify/smtp-port': {'default': 25, 'return_type': int}, | ||
'notify/from': {'default': '[email protected]', 'return_type': str}, | ||
'influxdb/org': {'default': 'pcw', 'return_type': str}, | ||
'influxdb/bucket': {'default': 'cloud_stat', 'return_type': str}, | ||
'influxdb/url': {'default': None, 'return_type': str}, | ||
} | ||
key = '/'.join([feature, property]) | ||
key = '/'.join([feature, feature_property]) | ||
if key not in default_values: | ||
raise LookupError(f"Missing {key} in default_values list") | ||
if namespace: | ||
setting = f'{feature}.namespace.{namespace}/{property}' | ||
setting = f'{feature}.namespace.{namespace}/{feature_property}' | ||
if PCWConfig.has(setting): | ||
return default_values[key]['return_type'](ConfigFile().get(setting)) | ||
return default_values[key]['return_type']( | ||
|
@@ -114,10 +117,10 @@ def has(setting: str) -> bool: | |
return False | ||
|
||
@staticmethod | ||
def getBoolean(config_path: str, namespace: str = None, default=False) -> bool: | ||
def getBoolean(config_path: str, namespace: str | None = None, default=False) -> bool: | ||
if namespace: | ||
feature, property = config_path.split('/') | ||
setting = f'{feature}.namespace.{namespace}/{property}' | ||
feature, feature_property = config_path.split('/') | ||
setting = f'{feature}.namespace.{namespace}/{feature_property}' | ||
if PCWConfig.has(setting): | ||
value = ConfigFile().get(setting) | ||
else: | ||
|