From 426da833f13df2b276d4ad11aa20ff3ad0c086bb Mon Sep 17 00:00:00 2001 From: Jack Shaw Date: Mon, 23 Oct 2023 11:42:04 +0100 Subject: [PATCH] Allow switch kwarg in refresh to switch to local charms Check if the switch param is pointing to a local charm. If it is, run local_refresh as if it were provided This is how the Juju CLI works, which was missing from pylibjuju. Replicate this. Also push url parsing of switch kwarg behind a check if it's local charm, meaning we avoid the possibility where we attempt to parse a path as if it were a url Resolves https://github.com/juju/python-libjuju/pull/967 --- juju/application.py | 27 ++++++++++++++------------- tests/integration/test_application.py | 15 +++++++++++++++ tests/unit/test_application.py | 11 +++++++++++ 3 files changed, 40 insertions(+), 13 deletions(-) diff --git a/juju/application.py b/juju/application.py index b530346c8..83cf933b7 100644 --- a/juju/application.py +++ b/juju/application.py @@ -4,7 +4,7 @@ import hashlib import json import logging -import pathlib +from pathlib import Path from . import model, tag, utils, jasyncio from .url import URL @@ -12,7 +12,7 @@ from .annotationhelper import _get_annotations, _set_annotations from .client import client from .errors import JujuError, JujuApplicationConfigError -from .bundle import get_charm_series +from .bundle import get_charm_series, is_local_charm from .placement import parse as parse_placement from .origin import Channel, Source @@ -661,6 +661,8 @@ async def refresh( :param str switch: Crossgrade charm url """ + if switch is not None and path is not None: + raise ValueError("switch and path are mutually exclusive") if switch is not None and revision is not None: raise ValueError("switch and revision are mutually exclusive") @@ -677,17 +679,18 @@ async def refresh( if charm_url_origin_result.error is not None: err = charm_url_origin_result.error raise JujuError(f'{err.code} : {err.message}') - charm_url = switch or charm_url_origin_result.url origin = charm_url_origin_result.charm_origin - if path is not None: + if path is not None or (switch is not None and is_local_charm(switch)): await self.local_refresh(origin, force, force_series, - force_units, path, resources) + force_units, path or switch, resources) return if resources is not None: raise NotImplementedError("resources option is not implemented") + # If switch is not None at this point, that means it's a switch to a store charm + charm_url = switch or charm_url_origin_result.url parsed_url = URL.parse(charm_url) charm_name = parsed_url.name @@ -733,8 +736,6 @@ async def refresh( err = charm_origin_result.error raise JujuError(f'{err.code} : {err.message}') - # Now take care of the resources: - # Already prepped the charm_resources # Now get the existing resources from the ResourcesFacade request_data = [client.Entity(self.tag)] @@ -808,22 +809,22 @@ async def local_refresh( path=None, resources=None): """Refresh the charm for this application with a local charm. - :param str channel: Channel to use when getting the charm from the - charm store, e.g. 'development' + :param dict charm_origin: The charm origin of the destination charm + we're refreshing to + :param bool force: Refresh even if validation checks fail :param bool force_series: Refresh even if series of deployed application is not supported by the new charm :param bool force_units: Refresh all units immediately, even if in error state :param str path: Refresh to a charm located at path :param dict resources: Dictionary of resource name/filepath pairs - :param int revision: Explicit refresh revision - :param str switch: Crossgrade charm url """ app_facade = self._facade() - if not isinstance(path, pathlib.Path): - path = pathlib.Path(path) + if isinstance(path, str) and path.startswith("local:"): + path = path[6:] + path = Path(path) charm_dir = path.expanduser().resolve() model_config = await self.get_config() diff --git a/tests/integration/test_application.py b/tests/integration/test_application.py index caee0cffe..424c5ff1a 100644 --- a/tests/integration/test_application.py +++ b/tests/integration/test_application.py @@ -219,6 +219,21 @@ async def test_upgrade_charm_resource_same_rev_no_update(event_loop): @base.bootstrapped +@pytest.mark.asyncio +async def test_refresh_charmhub_to_local(event_loop): + charm_path = INTEGRATION_TEST_DIR / 'charm' + async with base.CleanModel() as model: + app = await model.deploy('ubuntu', application_name='ubu-path') + await app.refresh(path=str(charm_path)) + assert app.data['charm-url'].startswith('local:') + + app = await model.deploy('ubuntu', application_name='ubu-switch') + await app.refresh(switch=str(charm_path)) + assert app.data['charm-url'].startswith('local:') + + +@base.bootstrapped +@pytest.mark.asyncio async def test_trusted(event_loop): async with base.CleanModel() as model: await model.deploy('ubuntu', trust=True) diff --git a/tests/unit/test_application.py b/tests/unit/test_application.py index cdc46060c..22cde8fdc 100644 --- a/tests/unit/test_application.py +++ b/tests/unit/test_application.py @@ -166,3 +166,14 @@ async def test_unexpose_endpoints_on_29_controller(self, mock_conn): application="panther", exposed_endpoints=["alpha", "beta"] ) + + +class TestRefreshApplication(asynctest.TestCase): + @asynctest.patch("juju.model.Model.connection") + async def test_refresh_mutually_exclusive_kwargs(self, mock_conn): + app = Application(entity_id="app-id", model=Model()) + with self.assertRaises(ValueError): + await app.refresh(switch="charm1", revision=10) + + with self.assertRaises(ValueError): + await app.refresh(switch="charm1", path="/path/to/charm2")