Skip to content

Commit

Permalink
Merge branch 'main' into wholesale-port-management-989
Browse files Browse the repository at this point in the history
  • Loading branch information
tonyandrewmeyer authored Sep 22, 2023
2 parents 685daf3 + ff6aeb9 commit b911855
Show file tree
Hide file tree
Showing 6 changed files with 96 additions and 103 deletions.
12 changes: 6 additions & 6 deletions ops/charm.py
Original file line number Diff line number Diff line change
Expand Up @@ -241,7 +241,7 @@ class ConfigChangedEvent(HookEvent):
This event notifies the charm of its initial configuration.
Typically, this event will fire between an :class:`install <InstallEvent>`
and a :class:`start <StartEvent>` during the startup sequence
(when you first deploy a unit), but in general it will fire whenever
(when a unit is first deployed), but in general it will fire whenever
the unit is (re)started, for example after pod churn on Kubernetes, on unit
rescheduling, on unit upgrade or refresh, and so on.
- As a specific instance of the above point: when networking changes
Expand Down Expand Up @@ -519,8 +519,8 @@ def snapshot(self) -> Dict[str, Any]:
def departing_unit(self) -> Optional[model.Unit]:
"""The :class:`ops.Unit` that is departing, if any.
You can use this to determine (for example) whether *you* are the
departing unit.
Use this method to determine (for example) whether this unit is the
departing one.
"""
# doing this on init would fail because `framework` gets patched in
# post-init
Expand Down Expand Up @@ -729,7 +729,7 @@ class SecretChangedEvent(SecretEvent):
a new secret revision, and all applications or units that are tracking this
secret will be notified via this event that a new revision is available.
Typically, you will want to fetch the new content by calling
Typically, the charm will fetch the new content by calling
:meth:`event.secret.get_content() <ops.Secret.get_content>` with ``refresh=True``
to tell Juju to start tracking the new revision.
"""
Expand Down Expand Up @@ -757,7 +757,7 @@ class SecretRemoveEvent(SecretEvent):
observers have updated to that new revision, this event will be fired to
inform the secret owner that the old revision can be removed.
Typically, you will want to call
Typically, the charm will call
:meth:`event.secret.remove_revision() <ops.Secret.remove_revision>` to
remove the now-unused revision.
"""
Expand Down Expand Up @@ -1005,7 +1005,7 @@ class CharmBase(Object):
:code:`CharmBase` is used to create a charm. This is done by inheriting
from :code:`CharmBase` and customising the subclass as required. So to
create your own charm, say ``MyCharm``, define a charm class and set up the
create a charm called ``MyCharm``, define a charm class and set up the
required event handlers (“hooks”) in its constructor::
import logging
Expand Down
24 changes: 11 additions & 13 deletions ops/framework.py
Original file line number Diff line number Diff line change
Expand Up @@ -202,7 +202,7 @@ def defer(self) -> None:
the result of an action, or any event other than metric events. The
queue of events will be dispatched before the new event is processed.
From the above you may deduce, but it's important to point out:
Important points that follow from the above:
* ``defer()`` does not interrupt the execution of the current event
handler. In almost all cases, a call to ``defer()`` should be followed
Expand All @@ -220,19 +220,18 @@ def defer(self) -> None:
The general desire to call ``defer()`` happens when some precondition
isn't yet met. However, care should be exercised as to whether it is
better to defer this event so that you see it again, or whether it is
better to defer this event so that it is seen again, or whether it is
better to just wait for the event that indicates the precondition has
been met.
For example, if ``config-changed`` is fired, and you are waiting for
different config, there is no reason to defer the event because there
will be a *different* ``config-changed`` event when the config actually
changes, rather than checking to see if maybe config has changed prior
to every other event that occurs.
For example, if handling a config change requires that two config
values are changed, there's no reason to defer the first
``config-changed`` because there will be a *second* ``config-changed``
event fired when the other config value changes.
Similarly, if you need 2 events to occur before you are ready to
proceed (say event A and B). When you see event A, you could chose to
``defer()`` it because you haven't seen B yet. However, that leads to:
Similarly, if two events need to occur before execution can proceed
(say event A and B), the event A handler could ``defer()`` because B
has not been seen yet. However, that leads to:
1. event A fires, calls defer()
Expand All @@ -242,7 +241,6 @@ def defer(self) -> None:
3. At some future time, event C happens, which also checks if A can
proceed.
"""
logger.debug("Deferring %s.", self)
self.deferred = True
Expand Down Expand Up @@ -1360,8 +1358,8 @@ def _from_iterable(cls, it: Iterable[_T]) -> Set[_T]:
Per https://docs.python.org/3/library/collections.abc.html
if the Set mixin is being used in a class with a different constructor signature,
you will need to override _from_iterable() with a classmethod that can construct
new instances from an iterable argument.
override _from_iterable() with a classmethod that can construct new instances
from an iterable argument.
"""
return set(it)

Expand Down
6 changes: 3 additions & 3 deletions ops/lib/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -96,9 +96,9 @@ def use(name: str, api: int, author: str) -> ModuleType:
def autoimport():
"""Find all libs in the path and enable use of them.
You only need to call this if you've installed a package or
otherwise changed sys.path in the current run, and need to see the
changes. Otherwise libraries are found on first call of `use`.
Call this function only when a package has been installed or sys.path has been
otherwise changed in the current run, and the changes need to be seen.
Otherwise libraries are found on first call of `use`.
DEPRECATED: This function is deprecated. Prefer charm libraries instead
(https://juju.is/docs/sdk/library).
Expand Down
4 changes: 2 additions & 2 deletions ops/main.py
Original file line number Diff line number Diff line change
Expand Up @@ -340,7 +340,7 @@ def is_restricted_context(self):

def _should_use_controller_storage(db_path: Path, meta: CharmMeta) -> bool:
"""Figure out whether we want to use controller storage or not."""
# if you've previously used local state, carry on using that
# if local state has been used previously, carry on using that
if db_path.exists():
return False

Expand Down Expand Up @@ -368,7 +368,7 @@ def main(charm_class: Type[ops.charm.CharmBase],
The event name is based on the way this executable was called (argv[0]).
Args:
charm_class: your charm class.
charm_class: the charm class to instantiate and receive the event.
use_juju_for_storage: whether to use controller-side storage. If not specified
then kubernetes charms that haven't previously used local storage and that
are running on a new enough Juju default to controller-side storage,
Expand Down
82 changes: 40 additions & 42 deletions ops/model.py
Original file line number Diff line number Diff line change
Expand Up @@ -123,7 +123,7 @@ def __init__(self, meta: 'ops.charm.CharmMeta', backend: '_ModelBackend'):

@property
def unit(self) -> 'Unit':
"""The unit that is running this code (that is, yourself).
"""The unit that is running this code.
Use :meth:`get_unit` to get an arbitrary unit by name.
"""
Expand Down Expand Up @@ -195,7 +195,7 @@ def uuid(self) -> str:
def get_unit(self, unit_name: str) -> 'Unit':
"""Get an arbitrary unit by name.
Use :attr:`unit` to get your own unit.
Use :attr:`unit` to get the current unit.
Internally this uses a cache, so asking for the same unit two times will
return the same object.
Expand All @@ -205,7 +205,7 @@ def get_unit(self, unit_name: str) -> 'Unit':
def get_app(self, app_name: str) -> 'Application':
"""Get an application by name.
Use :attr:`app` to get your own application.
Use :attr:`app` to get this charm's application.
Internally this uses a cache, so asking for the same application two times will
return the same object.
Expand Down Expand Up @@ -305,8 +305,8 @@ def get(self, entity_type: 'UnitOrApplicationType', name: str):
class Application:
"""Represents a named application in the model.
This might be your application, or might be an application that you are related to.
Charmers should not instantiate Application objects directly, but should use
This might be this charm's application, or might be an application this charm is related
to. Charmers should not instantiate Application objects directly, but should use
:attr:`Model.app` to get the application this unit is part of, or
:meth:`Model.get_app` if they need a reference to a given application.
"""
Expand Down Expand Up @@ -336,14 +336,13 @@ def status(self) -> 'StatusBase':
The status of remote units is always Unknown.
You can also use the :attr:`collect_app_status <CharmEvents.collect_app_status>`
event if you want to evaluate and set application status consistently
at the end of every hook.
Alternatively, use the :attr:`collect_app_status <CharmEvents.collect_app_status>`
event to evaluate and set application status consistently at the end of every hook.
Raises:
RuntimeError: if you try to set the status of another application, or if you try to
set the status of this application as a unit that is not the leader.
InvalidStatusError: if you try to set the status to something that is not a
RuntimeError: if setting the status of another application, or if setting the
status of this application as a unit that is not the leader.
InvalidStatusError: if setting the status to something that is not a
:class:`StatusBase`
Example::
Expand Down Expand Up @@ -453,8 +452,8 @@ def _calculate_expiry(expire: Optional[Union[datetime.datetime, datetime.timedel
class Unit:
"""Represents a named unit in the model.
This might be your unit, another unit of your application, or a unit of another application
that you are related to.
This might be the current unit, another unit of the charm's application, or a unit of
another application that the charm is related to.
"""

name: str
Expand Down Expand Up @@ -487,15 +486,14 @@ def _invalidate(self):
def status(self) -> 'StatusBase':
"""Used to report or read the status of a specific unit.
The status of any unit other than yourself is always Unknown.
The status of any unit other than the current unit is always Unknown.
You can also use the :attr:`collect_unit_status <CharmEvents.collect_unit_status>`
event if you want to evaluate and set unit status consistently at the
end of every hook.
Alternatively, use the :attr:`collect_unit_status <CharmEvents.collect_unit_status>`
event to evaluate and set unit status consistently at the end of every hook.
Raises:
RuntimeError: if you try to set the status of a unit other than yourself.
InvalidStatusError: if you try to set the status to something other than
RuntimeError: if setting the status of a unit other than the current unit
InvalidStatusError: if setting the status to something other than
a :class:`StatusBase`
Example::
Expand Down Expand Up @@ -531,10 +529,10 @@ def __repr__(self):
def is_leader(self) -> bool:
"""Return whether this unit is the leader of its application.
This can only be called for your own unit.
This can only be called for the current unit.
Raises:
RuntimeError: if called for a unit that is not yourself
RuntimeError: if called for another unit
"""
if self._is_our_unit:
# This value is not cached as it is not guaranteed to persist for the whole duration
Expand Down Expand Up @@ -900,8 +898,8 @@ class Network:

interfaces: List['NetworkInterface']
"""A list of network interface details. This includes the information
about how your application should be configured (for example, what IP
addresses you should bind to).
about how the application should be configured (for example, what IP
addresses should be bound to).
Multiple addresses for a single interface are represented as multiple
interfaces, for example::
Expand All @@ -910,11 +908,11 @@ class Network:
"""

ingress_addresses: List[Union[ipaddress.IPv4Address, ipaddress.IPv6Address, str]]
"""A list of IP addresses that other units should use to get in touch with you."""
"""A list of IP addresses that other units should use to get in touch with the charm."""

egress_subnets: List[Union[ipaddress.IPv4Network, ipaddress.IPv6Network]]
"""A list of networks representing the subnets that other units will see
you connecting from. Due to things like NAT it isn't always possible to
the charm connecting from. Due to things like NAT it isn't always possible to
narrow it down to a single address, but when it is clear, the CIDRs will
be constrained to a single address (for example, 10.0.0.1/32).
"""
Expand Down Expand Up @@ -945,10 +943,10 @@ def __init__(self, network_info: '_NetworkDict'):

@property
def bind_address(self) -> Optional[Union[ipaddress.IPv4Address, ipaddress.IPv6Address, str]]:
"""A single address that your application should bind() to.
"""A single address that the charm's application should bind() to.
For the common case where there is a single answer. This represents a single
address from :attr:`.interfaces` that can be used to configure where your
address from :attr:`.interfaces` that can be used to configure where the charm's
application should bind() and listen().
"""
if self.interfaces:
Expand All @@ -959,11 +957,11 @@ def bind_address(self) -> Optional[Union[ipaddress.IPv4Address, ipaddress.IPv6Ad
@property
def ingress_address(self) -> Optional[
Union[ipaddress.IPv4Address, ipaddress.IPv6Address, str]]:
"""The address other applications should use to connect to your unit.
"""The address other applications should use to connect to the current unit.
Due to things like public/private addresses, NAT and tunneling, the address you bind()
to is not always the address other people can use to connect() to you.
This is just the first address from :attr:`.ingress_addresses`.
Due to things like public/private addresses, NAT and tunneling, the address the charm
will bind() to is not always the address other people can use to connect() to the
charm. This is just the first address from :attr:`.ingress_addresses`.
"""
if self.ingress_addresses:
return self.ingress_addresses[0]
Expand Down Expand Up @@ -1153,12 +1151,12 @@ def id(self) -> Optional[str]:
may not include the model UUID (for cross-model secrets).
Charms should treat this as an opaque string for looking up secrets
and sharing them via relation data. If you need a charm-local "name"
and sharing them via relation data. If a charm-local "name" is needed
for a secret, use a :attr:`label`. (If a charm needs a truly unique
identifier for identifying one secret in a set of secrets of arbitrary
size, use :attr:`unique_identifier` -- this should be rare.)
This will be None if you obtained the secret using
This will be None if the secret was obtained using
:meth:`Model.get_secret` with a label but no ID.
"""
return self._id
Expand All @@ -1177,7 +1175,7 @@ def unique_identifier(self) -> Optional[str]:
cases where the charm has a set of secrets of arbitrary size, for
example, a group of 10 or 20 TLS certificates.
This will be None if you obtained the secret using
This will be None if the secret was obtained using
:meth:`Model.get_secret` with a label but no ID.
"""
if self._id is None:
Expand Down Expand Up @@ -1221,7 +1219,7 @@ def _on_secret_changed(self, event):
Juju will ensure that the entity (the owner or observer) only has one
secret with this label at once.
This will be None if you obtained the secret using
This will be None if the secret was obtained using
:meth:`Model.get_secret` with an ID but no label.
"""
return self._label
Expand Down Expand Up @@ -2227,7 +2225,7 @@ def push_path(self,
* /foo/foobar.txt
* /quux.txt
You could push the following ways::
These are various push examples::
# copy one file
container.push_path('/foo/foobar.txt', '/dst')
Expand Down Expand Up @@ -2308,7 +2306,7 @@ def pull_path(self,
* /foo/foobar.txt
* /quux.txt
You could pull the following ways::
These are various pull examples::
# copy one file
container.pull_path('/foo/foobar.txt', '/dst')
Expand Down Expand Up @@ -2699,9 +2697,9 @@ def __init__(self, relation_name: str, num_related: int, max_supported: int):
class RelationDataError(ModelError):
"""Raised when a relation data read/write is invalid.
This is raised if you're either trying to set a value to something that isn't a string,
or if you are trying to set a value in a bucket that you don't have access to. (For example,
another application/unit, or setting your application data without being the leader.)
This is raised either when trying to set a value to something that isn't a string,
or when trying to set a value in a bucket without the required access. (For example,
another application/unit, or setting application data without being the leader.)
"""


Expand All @@ -2710,9 +2708,9 @@ class RelationDataTypeError(RelationDataError):


class RelationDataAccessError(RelationDataError):
"""Raised by ``Relation.data[entity][key] = value`` if you don't have access.
"""Raised by ``Relation.data[entity][key] = value`` if unable to access.
This typically means that you don't have permission to write read/write the databag,
This typically means that permission to write read/write the databag is missing,
but in some cases it is raised when attempting to read/write from a deceased remote entity.
"""

Expand Down
Loading

0 comments on commit b911855

Please sign in to comment.