Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: expose the ability to set Pebble log targets #1074

Merged
1 change: 1 addition & 0 deletions CHANGES.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
# 2.9.0

* Added log target support to `ops.pebble` layers and plans.
* Added `Harness.run_action()`, `testing.ActionOutput`, and `testing.ActionFailed`

# 2.8.0
Expand Down
74 changes: 72 additions & 2 deletions ops/pebble.py
Original file line number Diff line number Diff line change
Expand Up @@ -124,16 +124,27 @@
'threshold': Optional[int]},
total=False)

# In Python 3.11+ 'services' and 'labels' should be NotRequired, and total=True.
LogTargetDict = typing.TypedDict('LogTargetDict',
{'override': Union[Literal['merge'], Literal['replace']],
'type': Literal['loki'],
'location': str,
'services': Optional[List[str]],
'labels': Optional[Dict[str, str]]},
total=False)

LayerDict = typing.TypedDict('LayerDict',
{'summary': str,
'description': str,
'services': Dict[str, ServiceDict],
'checks': Dict[str, CheckDict]},
'checks': Dict[str, CheckDict],
'log-targets': Dict[str, LogTargetDict]},
total=False)

PlanDict = typing.TypedDict('PlanDict',
{'services': Dict[str, ServiceDict],
'checks': Dict[str, CheckDict]},
'checks': Dict[str, CheckDict],
'log-targets': Dict[str, LogTargetDict]},
total=False)

_AuthDict = TypedDict('_AuthDict',
Expand Down Expand Up @@ -718,6 +729,9 @@ def __init__(self, raw: str):
for name, service in d.get('services', {}).items()}
self._checks: Dict[str, Check] = {name: Check(name, check)
for name, check in d.get('checks', {}).items()}
self._log_targets: Dict[str, LogTarget] = {
name: LogTarget(name, target)
for name, target in d.get('log-targets', {}).items()}

@property
def services(self) -> Dict[str, 'Service']:
Expand All @@ -735,11 +749,20 @@ def checks(self) -> Dict[str, 'Check']:
"""
return self._checks

@property
def log_targets(self) -> Dict[str, 'LogTarget']:
"""This plan's log targets mapping (maps log targer name to :class:`LogTarget`).
tonyandrewmeyer marked this conversation as resolved.
Show resolved Hide resolved

This property is currently read-only.
"""
return self._log_targets

def to_dict(self) -> 'PlanDict':
"""Convert this plan to its dict representation."""
fields = [
('services', {name: service.to_dict() for name, service in self._services.items()}),
('checks', {name: check.to_dict() for name, check in self._checks.items()}),
('log-targets', {name: target.to_dict() for name, target in self._log_targets.items()})
]
dct = {name: value for name, value in fields if value}
return typing.cast('PlanDict', dct)
Expand All @@ -766,6 +789,8 @@ class Layer:
services: Dict[str, 'Service']
#: Mapping of check to :class:`Check` defined by this layer.
checks: Dict[str, 'Check']
#: Mapping of target to :class:`LogTarget` defined by this layer.
log_targets: Dict[str, 'LogTarget']

def __init__(self, raw: Optional[Union[str, 'LayerDict']] = None):
if isinstance(raw, str):
Expand All @@ -780,6 +805,8 @@ def __init__(self, raw: Optional[Union[str, 'LayerDict']] = None):
for name, service in d.get('services', {}).items()}
self.checks = {name: Check(name, check)
for name, check in d.get('checks', {}).items()}
self.log_targets = {name: LogTarget(name, target)
for name, target in d.get('log-targets', {}).items()}

def to_yaml(self) -> str:
"""Convert this layer to its YAML representation."""
Expand All @@ -792,6 +819,7 @@ def to_dict(self) -> 'LayerDict':
('description', self.description),
('services', {name: service.to_dict() for name, service in self.services.items()}),
('checks', {name: check.to_dict() for name, check in self.checks.items()}),
('log-targets', {name: target.to_dict() for name, target in self.log_targets.items()})
]
dct = {name: value for name, value in fields if value}
return typing.cast('LayerDict', dct)
Expand Down Expand Up @@ -1028,6 +1056,48 @@ class CheckStatus(enum.Enum):
DOWN = 'down'


class LogTarget:
"""Represents a log target in a Pebble configuration layer."""

def __init__(self, name: str, raw: Optional['LogTargetDict'] = None):
self.name = name
dct: LogTargetDict = raw or {}
self.override: str = dct.get('override', '')
self.type_ = dct.get('type', '')
tonyandrewmeyer marked this conversation as resolved.
Show resolved Hide resolved
self.location = dct.get('location', '')
services = dct.get('services')
if services:
services = services[:]
tonyandrewmeyer marked this conversation as resolved.
Show resolved Hide resolved
self.services: Optional[List[str]] = services
labels = dct.get('labels')
if labels is not None:
labels = copy.deepcopy(labels)
self.labels: Optional[Dict[str, str]] = labels

def to_dict(self) -> 'LogTargetDict':
"""Convert this log target object to its dict representation."""
fields = [
('override', self.override),
('type', self.type_),
('location', self.location),
('services', self.services),
('labels', self.labels),
]
dct = {name: value for name, value in fields if value}
return typing.cast('LogTargetDict', dct)

def __repr__(self):
return f'LogTarget({self.to_dict()!r})'

def __eq__(self, other: Union['LogTargetDict', 'LogTarget']):
if isinstance(other, dict):
return self.to_dict() == other
elif isinstance(other, LogTarget):
return self.to_dict() == other.to_dict()
else:
return NotImplemented


class FileType(enum.Enum):
"""Enum of file types."""

Expand Down
112 changes: 112 additions & 0 deletions test/test_pebble.py
Original file line number Diff line number Diff line change
Expand Up @@ -479,6 +479,25 @@ def test_checks(self):
with self.assertRaises(AttributeError):
plan.checks = {} # type: ignore

def test_log_targets(self):
plan = pebble.Plan('')
self.assertEqual(plan.log_targets, {})

location = "https://example.com:3100/loki/api/v1/push"
plan = pebble.Plan(
'log-targets:\n baz:\n override: replace\n type: loki\n '
f'location: {location}')
tonyandrewmeyer marked this conversation as resolved.
Show resolved Hide resolved

self.assertEqual(len(plan.log_targets), 1)
self.assertEqual(plan.log_targets['baz'].name, 'baz')
self.assertEqual(plan.log_targets['baz'].override, 'replace')
self.assertEqual(plan.log_targets['baz'].type_, "loki")
self.assertEqual(plan.log_targets['baz'].location, location)

# Should be read-only ("can't set attribute")
with self.assertRaises(AttributeError):
plan.log_targets = {} # type: ignore

def test_yaml(self):
# Starting with nothing, we get the empty result
plan = pebble.Plan('')
Expand All @@ -496,6 +515,12 @@ def test_yaml(self):
bar:
http:
https://example.com/

log-targets:
baz:
override: replace
type: loki
location: https://example.com:3100/loki/api/v1/push
'''
plan = pebble.Plan(raw)
reformed = yaml.safe_dump(yaml.safe_load(raw))
Expand Down Expand Up @@ -524,6 +549,8 @@ def _assert_empty(self, layer: pebble.Layer):
self.assertEqual(layer.summary, '')
self.assertEqual(layer.description, '')
self.assertEqual(layer.services, {})
self.assertEqual(layer.checks, {})
self.assertEqual(layer.log_targets, {})
self.assertEqual(layer.to_dict(), {})

def test_no_args(self):
Expand All @@ -546,6 +573,17 @@ def test_dict(self):
'summary': 'Bar',
'command': 'echo bar',
},
},
'log-targets': {
'baz': {
'override': 'merge',
'type': 'loki',
'location': 'https://example.com',
'services': ['foo'],
'labels': {
'key': 'value $VAR',
}
},
}
}
s = pebble.Layer(d)
Expand All @@ -557,6 +595,12 @@ def test_dict(self):
self.assertEqual(s.services['bar'].name, 'bar')
self.assertEqual(s.services['bar'].summary, 'Bar')
self.assertEqual(s.services['bar'].command, 'echo bar')
self.assertEqual(s.log_targets['baz'].name, 'baz')
self.assertEqual(s.log_targets['baz'].override, 'merge')
self.assertEqual(s.log_targets['baz'].type_, 'loki')
self.assertEqual(s.log_targets['baz'].location, 'https://example.com')
self.assertEqual(s.log_targets['baz'].services, ['foo'])
self.assertEqual(s.log_targets['baz'].labels, {'key': 'value $VAR'})

self.assertEqual(s.to_dict(), d)

Expand All @@ -569,6 +613,11 @@ def test_yaml(self):
http:
url: https://example.com/
description: The quick brown fox!
log-targets:
baz:
location: https://example.com:3100
override: replace
type: loki
services:
bar:
command: echo bar
Expand Down Expand Up @@ -604,6 +653,10 @@ def test_yaml(self):
self.assertEqual(s.checks['chk'].name, 'chk')
self.assertEqual(s.checks['chk'].http, {'url': 'https://example.com/'})

self.assertEqual(s.log_targets['baz'].name, 'baz')
self.assertEqual(s.log_targets['baz'].override, 'replace')
self.assertEqual(s.log_targets['baz'].location, 'https://example.com:3100')

self.assertEqual(s.to_yaml(), yaml)
self.assertEqual(str(s), yaml)

Expand Down Expand Up @@ -884,6 +937,65 @@ def test_equality(self):
self.assertNotEqual(one, 5)


class TestLogTarget(unittest.TestCase):
def _assert_empty(self, target: pebble.LogTarget, name: str):
self.assertEqual(target.name, name)
self.assertEqual(target.override, '')
self.assertEqual(target.type_, '')
self.assertEqual(target.location, '')
self.assertIs(target.services, None)
self.assertIs(target.labels, None)

def test_name_only(self):
target = pebble.LogTarget('tgt')
self._assert_empty(target, 'tgt')

def test_dict(self):
d: pebble.LogTargetDict = {
'override': 'replace',
'type': 'loki',
'location': 'https://example.com:3100/loki/api/v1/push',
'services': ['+all'],
'labels': {'key': 'val', 'key2': 'val2'}
}
target = pebble.LogTarget('tgt', d)
self.assertEqual(target.name, 'tgt')
self.assertEqual(target.override, 'replace')
self.assertEqual(target.type_, 'loki')
self.assertEqual(target.location, 'https://example.com:3100/loki/api/v1/push')
self.assertEqual(target.services, ['+all'])
self.assertEqual(target.labels, {'key': 'val', 'key2': 'val2'})

self.assertEqual(target.to_dict(), d)

# Ensure pebble.Target has made copies of mutable objects.
assert target.services is not None and target.labels is not None
target.services[0] = '-all'
self.assertEqual(d['services'], ['+all'])
target.labels['key'] = 'val3'
assert d['labels'] is not None
self.assertEqual(d['labels']['key'], 'val')

def test_equality(self):
d: pebble.LogTargetDict = {
'override': 'replace',
'type': 'loki',
'location': 'https://example.com',
'services': ['foo', 'bar'],
'labels': {'k': 'v'}
}
one = pebble.LogTarget('one', d)
two = pebble.LogTarget('two', d)
self.assertEqual(one, two)
self.assertEqual(one, d)
self.assertEqual(two, d)
self.assertEqual(one, one.to_dict())
self.assertEqual(two, two.to_dict())
d['override'] = 'merge'
self.assertNotEqual(one, d)
self.assertNotEqual(one, 5)


class TestServiceInfo(unittest.TestCase):
def test_service_startup(self):
self.assertEqual(list(pebble.ServiceStartup), [
Expand Down
Loading