Skip to content

Commit

Permalink
Block snap management from refreshing
Browse files Browse the repository at this point in the history
  • Loading branch information
addyess committed Nov 14, 2024
1 parent b28dc12 commit 0b216e3
Show file tree
Hide file tree
Showing 2 changed files with 106 additions and 6 deletions.
41 changes: 38 additions & 3 deletions charms/worker/k8s/src/snap.py
Original file line number Diff line number Diff line change
Expand Up @@ -253,16 +253,51 @@ def management(charm: ops.CharmBase) -> None:
cache = snap_lib.SnapCache()
for args in _parse_management_arguments(charm):
which = cache[args.name]
install_args = args.dict(exclude_none=True)
if block_refresh(which, args):
continue
if isinstance(args, SnapFileArgument) and which.revision != "x1":
snap_lib.install_local(**args.dict(exclude_none=True))
snap_lib.install_local(**install_args)
elif isinstance(args, SnapStoreArgument) and args.revision:
if which.revision != args.revision:
log.info("Ensuring %s snap revision=%s", args.name, args.revision)
which.ensure(**args.dict(exclude_none=True))
which.ensure(**install_args)
which.hold()
elif isinstance(args, SnapStoreArgument):
log.info("Ensuring %s snap channel=%s", args.name, args.channel)
which.ensure(**args.dict(exclude_none=True))
which.ensure(**install_args)


def block_refresh(which: snap_lib.Snap, args: SnapArgument) -> bool:
"""Block snap refreshes if the snap is in a specific state.
Arguments:
which: The snap to check
args: The snap arguments
Returns:
bool: True if the snap should be blocked from refreshing
"""
if snap_lib.SnapState(which.state) == snap_lib.SnapState.Available:
log.info("Allowing %s snap installation", args.name)
return False
if _overridden_snap_installation().exists():
log.info("Allowing %s snap refresh due to snap installation override", args.name)
return False
if isinstance(args, SnapStoreArgument) and args.revision:
if block := which.revision != args.revision:
log.info("Blocking %s snap refresh to revision=%s", args.name, args.revision)
else:
log.info("Allowing %s snap refresh to same revision", args.name)
return block
if isinstance(args, SnapStoreArgument):
if block := which.channel != args.channel:
log.info("Blocking %s snap refresh to channel=%s", args.name, args.channel)
else:
log.info("Allowing %s snap refresh to same channel (%s)", args.name, args.channel)
return block
log.info("Blocking %s snap refresh", args.name)
return True


def version(snap: str) -> Tuple[Optional[str], bool]:
Expand Down
71 changes: 68 additions & 3 deletions charms/worker/k8s/tests/unit/test_snap.py
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,71 @@ def resource_snap_installation(tmp_path):
yield mock_path


@pytest.fixture()
def block_refresh():
"""Block snap refresh."""
with mock.patch("snap.block_refresh") as mocked:
mocked.return_value = False
yield mocked

@mock.patch("snap.snap_lib.SnapCache")
@pytest.mark.parametrize("state, as_file", [
[("present", "1234", None), False],
[("present", None, "edge"), False],
[("present", None, None), True],
], ids=[
"installed & store-by-channel",
"installed & store-by-revision",
"installed & file-without-override"
])
def test_block_refresh(cache, state, as_file, caplog, resource_snap_installation):
"""Test block refresh."""
caplog.set_level(0)
k8s_snap = cache()["k8s"]
k8s_snap.state, k8s_snap.revision, k8s_snap.channel = state
if as_file:
args = snap.SnapFileArgument(
name="k8s",
filename=resource_snap_installation.parent / "k8s_1234.snap",
)
else:
args = snap.SnapStoreArgument(
name="k8s",
channel="beta" if k8s_snap.channel else None,
revision="5678" if k8s_snap.revision else None,
)
assert snap.block_refresh(k8s_snap, args)
assert "Blocking k8s snap refresh" in caplog.text


@mock.patch("snap.snap_lib.SnapCache")
@pytest.mark.parametrize("state, overridden", [
[("available", None, None), None],
[("present", "1234", None), None],
[("present", None, "edge"), None],
[("present", None, None), True],
], ids=[
"not installed yet",
"installed & store-by-same-channel",
"installed & store-by-same-revision",
"installed & override"
])
def test_not_block_refresh(cache, state, overridden, caplog, resource_snap_installation):
"""Test block refresh."""
caplog.set_level(0)
k8s_snap = cache()["k8s"]
k8s_snap.state, k8s_snap.revision, k8s_snap.channel = state
if overridden:
resource_snap_installation.write_text("amd64:\n- install-type: store\n name: k8s\n channel: edge")
args = snap.SnapStoreArgument(
name="k8s",
channel=k8s_snap.channel,
revision=k8s_snap.revision,
)
assert not snap.block_refresh(k8s_snap, args)
assert "Allowing k8s snap" in caplog.text


@pytest.mark.usefixtures("missing_snap_installation")
def test_parse_no_file(harness):
"""Test no file exists."""
Expand Down Expand Up @@ -225,7 +290,7 @@ def test_parse_valid_file(mock_checkoutput, snap_installation, harness):
@mock.patch("snap._parse_management_arguments")
@mock.patch("snap.snap_lib.install_local")
@mock.patch("snap.snap_lib.SnapCache")
def test_management_installs_local(cache, install_local, args, harness):
def test_management_installs_local(cache, install_local, args, block_refresh, harness):
"""Test installer uses local installer."""
k8s_snap = cache()["k8s"]
args.return_value = [snap.SnapFileArgument(name="k8s", filename=Path("path/to/thing"))]
Expand All @@ -238,7 +303,7 @@ def test_management_installs_local(cache, install_local, args, harness):
@mock.patch("snap.snap_lib.install_local")
@mock.patch("snap.snap_lib.SnapCache")
@pytest.mark.parametrize("revision", [None, "123"])
def test_management_installs_store_from_channel(cache, install_local, args, revision, harness):
def test_management_installs_store_from_channel(cache, install_local, args, revision, block_refresh, harness):
"""Test installer uses store installer."""
k8s_snap = cache()["k8s"]
k8s_snap.revision = revision
Expand All @@ -252,7 +317,7 @@ def test_management_installs_store_from_channel(cache, install_local, args, revi
@mock.patch("snap.snap_lib.install_local")
@mock.patch("snap.snap_lib.SnapCache")
@pytest.mark.parametrize("revision", [None, "456", "123"])
def test_management_installs_store_from_revision(cache, install_local, args, revision, harness):
def test_management_installs_store_from_revision(cache, install_local, args, revision, block_refresh, harness):
"""Test installer uses store installer."""
k8s_snap = cache()["k8s"]
k8s_snap.revision = revision
Expand Down

0 comments on commit 0b216e3

Please sign in to comment.