Skip to content

Commit

Permalink
Merge pull request #145 from SUSE-Enceladus/max-bytes
Browse files Browse the repository at this point in the history
Add archive bytes limit option to config
  • Loading branch information
smarlowucf authored Jan 11, 2024
2 parents 7bf759b + 91b0e70 commit 549a18a
Show file tree
Hide file tree
Showing 14 changed files with 144 additions and 10 deletions.
45 changes: 36 additions & 9 deletions csp_billing_adapter/archive.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,24 +18,26 @@

import functools
import logging
import json

from csp_billing_adapter.config import Config
from csp_billing_adapter.utils import retry_on_exception

log = logging.getLogger('CSPBillingAdapter')

DEFAULT_RETENTION_PERIOD = 6 # in months


def append_metering_records(
archive: list,
billing_record: dict,
max_length: int
max_length: int,
max_bytes: int
) -> list:
"""
Append usage and metering records to the archive
If archive is larger than max length, drop the oldest record.
If archive is larger than max length drop the oldest
record. If the archive is larger than the max bytes
limit trim it until it satisfies the limit.
:param archive:
The list of meterings and usage records to append to.
Expand All @@ -44,17 +46,34 @@ def append_metering_records(
metering and usage records to be archived.
:param max_length:
The max size of the archive list.
:param max_bytes:
The max size in bytes of the archive.
:return:
The archive of meterings and usage records with the
billing_record appended. If archive ends up greater
than max lengh the first (oldest) record is dropped.
than max lengh or max bytes the archive is trimmed
as necessary to satisfy both max_length and
max_bytes.
"""
archive.append(billing_record)

if len(archive) > max_length:
return archive[1:]
else:
return archive
archive = archive[1:]

if max_bytes > 1:
# Treat 0 and 1 the same. Disable max bytes option.
# This prevents infitite loop when value is 1 since
# empty list is 2 bytes.
while True:
# Trim archive until it is smaller than max bytes
archive_size = len(bytes(json.dumps(archive), 'utf-8'))

if archive_size > max_bytes:
archive = archive[1:]
else:
break

return archive


def archive_record(
Expand All @@ -73,6 +92,13 @@ def archive_record(
The dictionary containing the most recent
metering and usage records to be archived.
"""
retention_period = config.archive_retention_period
bytes_limit = config.archive_bytes_limit

if retention_period < 1 or bytes_limit in (0, 1):
# Archive feature has been disabled, do nothing
return

archive = retry_on_exception(
functools.partial(
hook.get_metering_archive,
Expand All @@ -88,7 +114,8 @@ def archive_record(
archive = append_metering_records(
archive,
billing_record,
config.archive_retention_period or DEFAULT_RETENTION_PERIOD
retention_period,
bytes_limit
)

retry_on_exception(
Expand Down
3 changes: 3 additions & 0 deletions examples/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,9 @@ reporting_interval:
archive_retention_period:
Sets the time in months that meterings and usage records are retained in the data archive.
archive_bytes_limit:
Sets the max size in bytes for the metering archive. The archive will be trimmed to stay below this limit.
usage_metrics:
{metric name}:
Expand Down
1 change: 1 addition & 0 deletions examples/csp-billing-adapter-example-Rancher_NV.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ query_interval: 3600
reporting_api_is_cumulative: CSP_DEPENDENT_SETTING
reporting_interval: CSP_DEPENDENT_SETTING
archive_retention_period: 6
archive_bytes_limit: 1048576
usage_metrics:
managed_node_count:
consumption_reporting: volume
Expand Down
1 change: 1 addition & 0 deletions examples/csp-billing-adapter-example-SUMA.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ reporting_api_is_cumulative: CSP_DEPENDENT_SETTING
reporting_interval: CSP_DEPENDENT_SETTING
support_info_collection_cmd: CSP_DEPENDENT_SETTING
archive_retention_period: 6
archive_bytes_limit: -1
usage_metrics:
ha_systems:
consumption_reporting: volume
Expand Down
1 change: 1 addition & 0 deletions tests/data/config_bad.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 6
archive_bytes_limit: -1
usage_metrics:
managed_node_count:
consumption_reporting: volume
Expand Down
1 change: 1 addition & 0 deletions tests/data/config_dimensions_gap.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 6
archive_bytes_limit: -1
usage_metrics:
managed_node_count:
consumption_reporting: volume
Expand Down
1 change: 1 addition & 0 deletions tests/data/config_dimensions_no_tail.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 6
archive_bytes_limit: -1
usage_metrics:
managed_node_count:
consumption_reporting: volume
Expand Down
1 change: 1 addition & 0 deletions tests/data/config_good_average.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 6
archive_bytes_limit: -1
usage_metrics:
managed_node_count:
consumption_reporting: volume
Expand Down
31 changes: 31 additions & 0 deletions tests/data/config_good_average_no_archive.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
billing_interval: monthly
product_code: foo
query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 0
archive_bytes_limit: -1
usage_metrics:
managed_node_count:
consumption_reporting: volume
dimensions:
- dimension: tier_1
max: 15
min: 0
- dimension: tier_2
max: 50
min: 16
- dimension: tier_3
max: 100
min: 51
- dimension: tier_4
max: 250
min: 101
- dimension: tier_5
max: 1000
min: 251
- dimension: tier_6
min: 1001
min_consumption: 5
usage_aggregation: average
version: 1.1.1
1 change: 1 addition & 0 deletions tests/data/config_good_maximum.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 6
archive_bytes_limit: -1
usage_metrics:
managed_node_count:
consumption_reporting: volume
Expand Down
1 change: 1 addition & 0 deletions tests/data/config_invalid_consumption.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 6
archive_bytes_limit: -1
usage_metrics:
managed_node_count:
consumption_reporting: invalid
Expand Down
1 change: 1 addition & 0 deletions tests/data/config_no_usage_metrics.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,4 +4,5 @@ query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 6
archive_bytes_limit: -1
version: 1.1.1
1 change: 1 addition & 0 deletions tests/data/config_testing_mixed.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ query_interval: 3600
reporting_api_is_cumulative: true
reporting_interval: 3600
archive_retention_period: 6
archive_bytes_limit: -1
usage_metrics:
jobs:
# test metric with average aggregation, volume reporting
Expand Down
65 changes: 64 additions & 1 deletion tests/unit/test_archive.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,8 @@
for the archive util functions.
"""

from pytest import mark

from csp_billing_adapter.archive import (
append_metering_records,
archive_record
Expand All @@ -39,12 +41,33 @@ def test_append_metering_records():
}

for i in range(10):
archive = append_metering_records(archive, records, 6)
archive = append_metering_records(archive, records, 6, 0)

assert len(archive) == 6
assert archive[4] == records


def test_append_metering_records_max_bytes():
archive = []
records = {
"bill_time": "2024-01-03T20:06:42.076972+00:00",
"metering_id": "FFFFFFFF-FFFF-FFFF-FFFF-FFFFFFFFFFFF",
"usage_records": [
{
"managed_node_count": 9,
"managed_systems": 6,
"reporting_time": "2024-01-03T20:06:42.076972+00:00"
}
]
}

for i in range(4):
archive = append_metering_records(archive, records, 6, 456)

assert len(archive) == 2
assert archive[0] == records


def test_archive_record(cba_pm, cba_config):
record = {
'billing_time': '2024-02-09T18:11:59.527064+00:00',
Expand Down Expand Up @@ -83,3 +106,43 @@ def test_archive_record(cba_pm, cba_config):
archive = cba_pm.hook.get_metering_archive(config=cba_config)
assert len(archive) == 1
assert archive[0] == record


@mark.config('config_good_average_no_archive.yaml')
def test_archive_disabled(cba_pm, cba_config):
record = {
'billing_time': '2024-02-09T18:11:59.527064+00:00',
'billing_status': {
'tier_1': {
'record_id': 'd92c6e6556b14770994f5b64ebe3d339',
'status': 'succeeded'
}
},
'billed_usage': {
'tier_1': 10
},
'usage_records': [
{
'managed_node_count': 9,
'reporting_time': '2024-01-09T18:11:59.527673+00:00',
'base_product': 'cpe:/o:suse:product:v1.2.3'
},
{
'managed_node_count': 9,
'reporting_time': '2024-01-09T18:11:59.529096+00:00',
'base_product': 'cpe:/o:suse:product:v1.2.3'
},
{
'managed_node_count': 10,
'reporting_time': '2024-01-09T18:11:59.531424+00:00',
'base_product': 'cpe:/o:suse:product:v1.2.3'
}
]
}
archive_record(
cba_pm.hook,
cba_config,
record
)
archive = cba_pm.hook.get_metering_archive(config=cba_config)
assert len(archive) == 0

0 comments on commit 549a18a

Please sign in to comment.