From 4790ab3b4be24e3ce68ead23c0dbc996f3e8a314 Mon Sep 17 00:00:00 2001 From: matusdrobuliak66 Date: Wed, 18 Dec 2024 16:40:51 +0100 Subject: [PATCH 1/7] release seats on issues --- ...ackground_task_periodic_heartbeat_check.py | 15 +- .../modules/db/licensed_items_checkouts_db.py | 43 ++++++ .../test_licensed_items_checkouts_db.py | 139 ++++++++++++++++++ 3 files changed, 194 insertions(+), 3 deletions(-) create mode 100644 services/resource-usage-tracker/tests/unit/with_dbs/test_licensed_items_checkouts_db.py diff --git a/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/background_task_periodic_heartbeat_check.py b/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/background_task_periodic_heartbeat_check.py index fba9332502e..4401198baa1 100644 --- a/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/background_task_periodic_heartbeat_check.py +++ b/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/background_task_periodic_heartbeat_check.py @@ -1,6 +1,6 @@ import asyncio import logging -from datetime import datetime, timedelta, timezone +from datetime import UTC, datetime, timedelta from fastapi import FastAPI from models_library.resource_tracker import ( @@ -15,7 +15,11 @@ from ..core.settings import ApplicationSettings from ..models.credit_transactions import CreditTransactionCreditsAndStatusUpdate from ..models.service_runs import ServiceRunStoppedAtUpdate -from .modules.db import credit_transactions_db, service_runs_db +from .modules.db import ( + credit_transactions_db, + licensed_items_checkouts_db, + service_runs_db, +) from .utils import compute_service_run_credit_costs, make_negative _logger = logging.getLogger(__name__) @@ -116,6 +120,11 @@ async def _close_unhealthy_service( db_engine, data=update_credit_transaction ) + # 3. Release license seats in case some were checked out but not properly released. + await licensed_items_checkouts_db.force_release_license_seats_by_run_id( + db_engine, service_run_id=service_run_id + ) + async def periodic_check_of_running_services_task(app: FastAPI) -> None: _logger.info("Periodic check started") @@ -124,7 +133,7 @@ async def periodic_check_of_running_services_task(app: FastAPI) -> None: app_settings: ApplicationSettings = app.state.settings _db_engine = app.state.engine - base_start_timestamp = datetime.now(tz=timezone.utc) + base_start_timestamp = datetime.now(tz=UTC) # Get all current running services (across all products) total_count: PositiveInt = await service_runs_db.total_service_runs_with_running_status_across_all_products( diff --git a/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/modules/db/licensed_items_checkouts_db.py b/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/modules/db/licensed_items_checkouts_db.py index 2402a8c52be..8df27f1f5c9 100644 --- a/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/modules/db/licensed_items_checkouts_db.py +++ b/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/modules/db/licensed_items_checkouts_db.py @@ -1,9 +1,11 @@ +import logging from datetime import datetime from typing import cast import sqlalchemy as sa from models_library.licensed_items import LicensedItemID from models_library.products import ProductName +from models_library.resource_tracker import ServiceRunId from models_library.resource_tracker_licensed_items_checkouts import ( LicensedItemCheckoutID, ) @@ -27,6 +29,9 @@ LicensedItemCheckoutDB, ) +_logger = logging.getLogger(__name__) + + _SELECTION_ARGS = ( resource_tracker_licensed_items_checkouts.c.licensed_item_checkout_id, resource_tracker_licensed_items_checkouts.c.licensed_item_id, @@ -214,3 +219,41 @@ async def get_currently_used_seats_for_item_and_wallet( if total_sum is None: return 0 return cast(int, total_sum) + + +async def force_release_license_seats_by_run_id( + engine: AsyncEngine, + connection: AsyncConnection | None = None, + *, + service_run_id: ServiceRunId, +) -> None: + """ + Purpose: This function is utilized by a periodic heartbeat check task that monitors whether running services are + sending heartbeat signals. If heartbeat signals are not received within a specified timeframe and a service is + deemed unhealthy, this function ensures the proper release of any licensed seats that were not correctly released by + the unhealthy service. + Currently, this functionality is primarily used to handle the release of a single seat allocated to the VIP model. + """ + update_stmt = ( + resource_tracker_licensed_items_checkouts.update() + .values( + modified=sa.func.now(), + stopped_at=sa.func.now(), + ) + .where( + ( + resource_tracker_licensed_items_checkouts.c.service_run_id + == service_run_id + ) + & (resource_tracker_licensed_items_checkouts.c.stopped_at.is_(None)) + ) + .returning(sa.literal_column("*")) + ) + + async with transaction_context(engine, connection) as conn: + result = await conn.execute(update_stmt) + released_seats = result.fetchall() + if released_seats: + _logger.error( + "Force release of %s seats: %s", len(released_seats), released_seats + ) diff --git a/services/resource-usage-tracker/tests/unit/with_dbs/test_licensed_items_checkouts_db.py b/services/resource-usage-tracker/tests/unit/with_dbs/test_licensed_items_checkouts_db.py new file mode 100644 index 00000000000..34e1e0f107b --- /dev/null +++ b/services/resource-usage-tracker/tests/unit/with_dbs/test_licensed_items_checkouts_db.py @@ -0,0 +1,139 @@ +# pylint:disable=unused-variable +# pylint:disable=unused-argument +# pylint:disable=redefined-outer-name +# pylint:disable=too-many-arguments + + +from datetime import UTC, datetime +from typing import Generator +from unittest import mock + +import pytest +import sqlalchemy as sa +from models_library.basic_types import IDStr +from models_library.rest_ordering import OrderBy +from simcore_postgres_database.models.resource_tracker_licensed_items_checkouts import ( + resource_tracker_licensed_items_checkouts, +) +from simcore_postgres_database.models.resource_tracker_service_runs import ( + resource_tracker_service_runs, +) +from simcore_service_resource_usage_tracker.models.licensed_items_checkouts import ( + CreateLicensedItemCheckoutDB, +) +from simcore_service_resource_usage_tracker.services.modules.db import ( + licensed_items_checkouts_db, +) + +pytest_simcore_core_services_selection = [ + "postgres", +] +pytest_simcore_ops_services_selection = [ + "adminer", +] + + +_USER_ID_1 = 1 +_WALLET_ID = 6 + + +@pytest.fixture() +def resource_tracker_service_run_id( + postgres_db: sa.engine.Engine, random_resource_tracker_service_run +) -> Generator[str, None, None]: + with postgres_db.connect() as con: + result = con.execute( + resource_tracker_service_runs.insert() + .values( + **random_resource_tracker_service_run( + user_id=_USER_ID_1, wallet_id=_WALLET_ID + ) + ) + .returning(resource_tracker_service_runs.c.service_run_id) + ) + row = result.first() + assert row + + yield row[0] + + con.execute(resource_tracker_licensed_items_checkouts.delete()) + con.execute(resource_tracker_service_runs.delete()) + + +async def test_licensed_items_checkouts_db__force_release_license_seats_by_run_id( + mocked_redis_server: None, + mocked_setup_rabbitmq: mock.Mock, + resource_tracker_service_run_id, + initialized_app, +): + engine = initialized_app.state.engine + + # SETUP + _create_license_item_checkout_db_1 = CreateLicensedItemCheckoutDB( + licensed_item_id="beb16d18-d57d-44aa-a638-9727fa4a72ef", + wallet_id=_WALLET_ID, + user_id=_USER_ID_1, + user_email="test@test.com", + product_name="osparc", + service_run_id=resource_tracker_service_run_id, # <-- RUN ID 1 + started_at=datetime.now(tz=UTC), + num_of_seats=1, + ) + await licensed_items_checkouts_db.create( + engine, data=_create_license_item_checkout_db_1 + ) + + _create_license_item_checkout_db_2 = _create_license_item_checkout_db_1.model_dump() + _create_license_item_checkout_db_2[ + "licensed_item_id" + ] = "b1b96583-333f-44d6-b1e0-5c0a8af555bf" + await licensed_items_checkouts_db.create( + engine, + data=CreateLicensedItemCheckoutDB.model_construct( + **_create_license_item_checkout_db_2 # <-- RUN ID 1 + ), + ) + + _create_license_item_checkout_db_3 = _create_license_item_checkout_db_1.model_dump() + _create_license_item_checkout_db_3[ + "licensed_item_id" + ] = "38a5ce59-876f-482a-ace1-d3b2636feac6" + checkout = await licensed_items_checkouts_db.create( + engine, + data=CreateLicensedItemCheckoutDB.model_construct( + **_create_license_item_checkout_db_3 # <-- RUN ID 1 + ), + ) + + _helper_time = datetime.now(UTC) + await licensed_items_checkouts_db.update( + engine, + licensed_item_checkout_id=checkout.licensed_item_checkout_id, + product_name="osparc", + stopped_at=_helper_time, + ) + + # TEST FORCE RELEASE LICENSE SEATS + await licensed_items_checkouts_db.force_release_license_seats_by_run_id( + engine, service_run_id=resource_tracker_service_run_id + ) + + # ASSERT + total, items = await licensed_items_checkouts_db.list_( + engine, + product_name="osparc", + filter_wallet_id=_WALLET_ID, + offset=0, + limit=5, + order_by=OrderBy(field=IDStr("started_at")), + ) + assert total == 3 + assert len(items) == 3 + + _helper_count = 0 + for item in items: + assert isinstance(item.stopped_at, datetime) + if item.stopped_at > _helper_time: + _helper_count += 1 + + assert _helper_count == 2 From ae14a78224d397c0f80f16887758e7973ccec75a Mon Sep 17 00:00:00 2001 From: matusdrobuliak66 Date: Wed, 18 Dec 2024 16:45:17 +0100 Subject: [PATCH 2/7] release seats on issues --- .../tests/unit/with_dbs/test_licensed_items_checkouts_db.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/services/resource-usage-tracker/tests/unit/with_dbs/test_licensed_items_checkouts_db.py b/services/resource-usage-tracker/tests/unit/with_dbs/test_licensed_items_checkouts_db.py index 34e1e0f107b..5f0fc5a1f5b 100644 --- a/services/resource-usage-tracker/tests/unit/with_dbs/test_licensed_items_checkouts_db.py +++ b/services/resource-usage-tracker/tests/unit/with_dbs/test_licensed_items_checkouts_db.py @@ -75,7 +75,7 @@ async def test_licensed_items_checkouts_db__force_release_license_seats_by_run_i user_id=_USER_ID_1, user_email="test@test.com", product_name="osparc", - service_run_id=resource_tracker_service_run_id, # <-- RUN ID 1 + service_run_id=resource_tracker_service_run_id, started_at=datetime.now(tz=UTC), num_of_seats=1, ) @@ -90,7 +90,7 @@ async def test_licensed_items_checkouts_db__force_release_license_seats_by_run_i await licensed_items_checkouts_db.create( engine, data=CreateLicensedItemCheckoutDB.model_construct( - **_create_license_item_checkout_db_2 # <-- RUN ID 1 + **_create_license_item_checkout_db_2 ), ) @@ -101,7 +101,7 @@ async def test_licensed_items_checkouts_db__force_release_license_seats_by_run_i checkout = await licensed_items_checkouts_db.create( engine, data=CreateLicensedItemCheckoutDB.model_construct( - **_create_license_item_checkout_db_3 # <-- RUN ID 1 + **_create_license_item_checkout_db_3 ), ) From 42071a0eeeb5ae0201a9270f3266b16ab447082b Mon Sep 17 00:00:00 2001 From: matusdrobuliak66 Date: Thu, 19 Dec 2024 11:20:48 +0100 Subject: [PATCH 3/7] add force release of license seats on stop event --- .../modules/db/licensed_items_checkouts_db.py | 4 ++-- .../services/process_message_running_service.py | 15 +++++++++++++-- 2 files changed, 15 insertions(+), 4 deletions(-) diff --git a/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/modules/db/licensed_items_checkouts_db.py b/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/modules/db/licensed_items_checkouts_db.py index 8df27f1f5c9..5035a637199 100644 --- a/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/modules/db/licensed_items_checkouts_db.py +++ b/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/modules/db/licensed_items_checkouts_db.py @@ -5,11 +5,11 @@ import sqlalchemy as sa from models_library.licensed_items import LicensedItemID from models_library.products import ProductName -from models_library.resource_tracker import ServiceRunId from models_library.resource_tracker_licensed_items_checkouts import ( LicensedItemCheckoutID, ) from models_library.rest_ordering import OrderBy, OrderDirection +from models_library.services_types import ServiceRunID from models_library.wallets import WalletID from pydantic import NonNegativeInt from servicelib.rabbitmq.rpc_interfaces.resource_usage_tracker.errors import ( @@ -225,7 +225,7 @@ async def force_release_license_seats_by_run_id( engine: AsyncEngine, connection: AsyncConnection | None = None, *, - service_run_id: ServiceRunId, + service_run_id: ServiceRunID, ) -> None: """ Purpose: This function is utilized by a periodic heartbeat check task that monitors whether running services are diff --git a/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/process_message_running_service.py b/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/process_message_running_service.py index e9234f65435..88553f51705 100644 --- a/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/process_message_running_service.py +++ b/services/resource-usage-tracker/src/simcore_service_resource_usage_tracker/services/process_message_running_service.py @@ -33,7 +33,12 @@ ServiceRunLastHeartbeatUpdate, ServiceRunStoppedAtUpdate, ) -from .modules.db import credit_transactions_db, pricing_plans_db, service_runs_db +from .modules.db import ( + credit_transactions_db, + licensed_items_checkouts_db, + pricing_plans_db, + service_runs_db, +) from .modules.rabbitmq import RabbitMQClient, get_rabbitmq_client from .utils import ( compute_service_run_credit_costs, @@ -269,9 +274,15 @@ async def _process_stop_event( running_service = await service_runs_db.update_service_run_stopped_at( db_engine, data=update_service_run_stopped_at ) + await licensed_items_checkouts_db.force_release_license_seats_by_run_id( + db_engine, service_run_id=msg.service_run_id + ) if running_service is None: - _logger.error("Nothing to update. This should not happen investigate.") + _logger.error( + "Nothing to update. This should not happen investigate. service_run_id: %s", + msg.service_run_id, + ) return if running_service.wallet_id and running_service.pricing_unit_cost is not None: From b9036d8f8702bf01b18610813c8cfb36de2e5183 Mon Sep 17 00:00:00 2001 From: matusdrobuliak66 Date: Fri, 20 Dec 2024 13:37:36 +0100 Subject: [PATCH 4/7] list/get checkouts --- .../web-server/_licensed_items_checkouts.py | 57 ++++++++ .../licensed_items_checkouts.py | 29 +++- .../webserver/licenses/licensed_items.py | 10 +- ...pi.py => _licensed_items_checkouts_api.py} | 100 ++++++++++++- .../_licensed_items_checkouts_handlers.py | 131 ++++++++++++++++++ .../_licensed_items_checkouts_models.py | 56 ++++++++ .../licenses/_licensed_items_purchases_api.py | 1 + .../licenses/_rpc.py | 58 +++++--- .../test_licensed_items_checkouts_handlers.py | 84 +++++++++++ .../test_licensed_items_purchases_handlers.py | 2 +- .../with_dbs/04/licenses/test_licenses_rpc.py | 8 +- 11 files changed, 498 insertions(+), 38 deletions(-) create mode 100644 api/specs/web-server/_licensed_items_checkouts.py rename services/web/server/src/simcore_service_webserver/licenses/{_licensed_checkouts_api.py => _licensed_items_checkouts_api.py} (55%) create mode 100644 services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py create mode 100644 services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_models.py create mode 100644 services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py diff --git a/api/specs/web-server/_licensed_items_checkouts.py b/api/specs/web-server/_licensed_items_checkouts.py new file mode 100644 index 00000000000..cfc51a7c424 --- /dev/null +++ b/api/specs/web-server/_licensed_items_checkouts.py @@ -0,0 +1,57 @@ +""" Helper script to generate OAS automatically +""" + +# pylint: disable=redefined-outer-name +# pylint: disable=unused-argument +# pylint: disable=unused-variable +# pylint: disable=too-many-arguments + +from typing import Annotated + +from _common import as_query +from fastapi import APIRouter, Depends +from models_library.api_schemas_webserver.licensed_items_purchases import ( + LicensedItemPurchaseGet, +) +from models_library.generics import Envelope +from models_library.rest_error import EnvelopedError +from models_library.rest_pagination import Page +from simcore_service_webserver._meta import API_VTAG +from simcore_service_webserver.licenses._exceptions_handlers import _TO_HTTP_ERROR_MAP +from simcore_service_webserver.licenses._licensed_items_checkouts_models import ( + LicensedItemCheckoutPathParams, + LicensedItemsCheckoutsListQueryParams, +) +from simcore_service_webserver.wallets._handlers import WalletsPathParams + +router = APIRouter( + prefix=f"/{API_VTAG}", + tags=[ + "licenses", + ], + responses={ + i.status_code: {"model": EnvelopedError} for i in _TO_HTTP_ERROR_MAP.values() + }, +) + + +@router.get( + "/wallets/{wallet_id}/licensed-items-checkouts", + response_model=Page[LicensedItemPurchaseGet], + tags=["wallets"], +) +async def list_licensed_item_checkouts_for_wallet( + _path: Annotated[WalletsPathParams, Depends()], + _query: Annotated[as_query(LicensedItemsCheckoutsListQueryParams), Depends()], +): + ... + + +@router.get( + "/licensed-items-checkouts/{licensed_item_checkout_id}", + response_model=Envelope[LicensedItemPurchaseGet], +) +async def get_licensed_item_checkout( + _path: Annotated[LicensedItemCheckoutPathParams, Depends()], +): + ... diff --git a/packages/models-library/src/models_library/api_schemas_webserver/licensed_items_checkouts.py b/packages/models-library/src/models_library/api_schemas_webserver/licensed_items_checkouts.py index a3ee122ddee..c8fd22ce581 100644 --- a/packages/models-library/src/models_library/api_schemas_webserver/licensed_items_checkouts.py +++ b/packages/models-library/src/models_library/api_schemas_webserver/licensed_items_checkouts.py @@ -1,7 +1,7 @@ from datetime import datetime from typing import NamedTuple -from pydantic import PositiveInt +from pydantic import BaseModel, PositiveInt from ..licensed_items import LicensedItemID from ..products import ProductName @@ -10,8 +10,29 @@ from ..wallets import WalletID from ._base import OutputSchema +# RPC -class LicensedItemCheckoutGet(OutputSchema): + +class LicensedItemCheckoutRpcGet(BaseModel): + licensed_item_checkout_id: LicensedItemCheckoutID + licensed_item_id: LicensedItemID + wallet_id: WalletID + user_id: UserID + product_name: ProductName + started_at: datetime + stopped_at: datetime | None + num_of_seats: int + + +class LicensedItemCheckoutRpcGetPage(NamedTuple): + items: list[LicensedItemCheckoutRpcGet] + total: PositiveInt + + +# Rest + + +class LicensedItemCheckoutRestGet(OutputSchema): licensed_item_checkout_id: LicensedItemCheckoutID licensed_item_id: LicensedItemID wallet_id: WalletID @@ -22,6 +43,6 @@ class LicensedItemCheckoutGet(OutputSchema): num_of_seats: int -class LicensedItemUsageGetPage(NamedTuple): - items: list[LicensedItemCheckoutGet] +class LicensedItemCheckoutRestGetPage(NamedTuple): + items: list[LicensedItemCheckoutRestGet] total: PositiveInt diff --git a/packages/service-library/src/servicelib/rabbitmq/rpc_interfaces/webserver/licenses/licensed_items.py b/packages/service-library/src/servicelib/rabbitmq/rpc_interfaces/webserver/licenses/licensed_items.py index f767882d247..0f86ab63d79 100644 --- a/packages/service-library/src/servicelib/rabbitmq/rpc_interfaces/webserver/licenses/licensed_items.py +++ b/packages/service-library/src/servicelib/rabbitmq/rpc_interfaces/webserver/licenses/licensed_items.py @@ -6,7 +6,7 @@ LicensedItemGetPage, ) from models_library.api_schemas_webserver.licensed_items_checkouts import ( - LicensedItemCheckoutGet, + LicensedItemCheckoutRpcGet, ) from models_library.licensed_items import LicensedItemID from models_library.products import ProductName @@ -78,7 +78,7 @@ async def checkout_licensed_item_for_wallet( licensed_item_id: LicensedItemID, num_of_seats: int, service_run_id: ServiceRunID, -) -> LicensedItemCheckoutGet: +) -> LicensedItemCheckoutRpcGet: result = await rabbitmq_rpc_client.request( WEBSERVER_RPC_NAMESPACE, TypeAdapter(RPCMethodName).validate_python("checkout_licensed_item_for_wallet"), @@ -89,7 +89,7 @@ async def checkout_licensed_item_for_wallet( num_of_seats=num_of_seats, service_run_id=service_run_id, ) - assert isinstance(result, LicensedItemCheckoutGet) # nosec + assert isinstance(result, LicensedItemCheckoutRpcGet) # nosec return result @@ -100,7 +100,7 @@ async def release_licensed_item_for_wallet( product_name: ProductName, user_id: UserID, licensed_item_checkout_id: LicensedItemCheckoutID, -) -> LicensedItemCheckoutGet: +) -> LicensedItemCheckoutRpcGet: result = await rabbitmq_rpc_client.request( WEBSERVER_RPC_NAMESPACE, TypeAdapter(RPCMethodName).validate_python("release_licensed_item_for_wallet"), @@ -108,5 +108,5 @@ async def release_licensed_item_for_wallet( user_id=user_id, licensed_item_checkout_id=licensed_item_checkout_id, ) - assert isinstance(result, LicensedItemCheckoutGet) # nosec + assert isinstance(result, LicensedItemCheckoutRpcGet) # nosec return result diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_checkouts_api.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_api.py similarity index 55% rename from services/web/server/src/simcore_service_webserver/licenses/_licensed_checkouts_api.py rename to services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_api.py index ba140d565d5..87a8aaf14c5 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_licensed_checkouts_api.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_api.py @@ -2,14 +2,12 @@ from models_library.api_schemas_resource_usage_tracker import ( licensed_items_checkouts as rut_licensed_items_checkouts, ) -from models_library.api_schemas_webserver import ( - licensed_items_checkouts as webserver_licensed_items_checkouts, -) from models_library.licensed_items import LicensedItemID from models_library.products import ProductName from models_library.resource_tracker_licensed_items_checkouts import ( LicensedItemCheckoutID, ) +from models_library.rest_ordering import OrderBy from models_library.services_types import ServiceRunID from models_library.users import UserID from models_library.wallets import WalletID @@ -20,6 +18,94 @@ from ..rabbitmq import get_rabbitmq_rpc_client from ..users.api import get_user from ..wallets.api import get_wallet_by_user +from ._licensed_items_checkouts_models import ( + LicensedItemCheckoutGet, + LicensedItemCheckoutGetPage, +) + + +async def list_licensed_items_checkouts_for_wallet( + app: web.Application, + *, + # access context + product_name: ProductName, + user_id: UserID, + wallet_id: WalletID, + offset: int, + limit: int, + order_by: OrderBy, +) -> LicensedItemCheckoutGetPage: + # Check whether user has access to the wallet + await get_wallet_by_user( + app, + user_id=user_id, + wallet_id=wallet_id, + product_name=product_name, + ) + + rpc_client = get_rabbitmq_rpc_client(app) + + result = await licensed_items_checkouts.get_licensed_items_checkouts_page( + rpc_client, + product_name=product_name, + filter_wallet_id=wallet_id, + offset=offset, + limit=limit, + order_by=order_by, + ) + + return LicensedItemCheckoutGetPage( + total=result.total, + items=[ + LicensedItemCheckoutGet.model_construct( + licensed_item_checkout_id=checkout_item.licensed_item_checkout_id, + licensed_item_id=checkout_item.licensed_item_id, + wallet_id=checkout_item.wallet_id, + user_id=checkout_item.user_id, + product_name=checkout_item.product_name, + started_at=checkout_item.started_at, + stopped_at=checkout_item.stopped_at, + num_of_seats=checkout_item.num_of_seats, + ) + for checkout_item in result.items + ], + ) + + +async def get_licensed_item_checkout( + app: web.Application, + *, + # access context + product_name: ProductName, + user_id: UserID, + licensed_item_checkout_id: LicensedItemCheckoutID, +) -> LicensedItemCheckoutGet: + rpc_client = get_rabbitmq_rpc_client(app) + + checkout_item = await licensed_items_checkouts.get_licensed_item_checkout( + rpc_client, + product_name=product_name, + licensed_item_checkout_id=licensed_item_checkout_id, + ) + + # Check whether user has access to the wallet + await get_wallet_by_user( + app, + user_id=user_id, + wallet_id=checkout_item.wallet_id, + product_name=product_name, + ) + + return LicensedItemCheckoutGet.model_construct( + licensed_item_checkout_id=checkout_item.licensed_item_checkout_id, + licensed_item_id=checkout_item.licensed_item_id, + wallet_id=checkout_item.wallet_id, + user_id=checkout_item.user_id, + product_name=checkout_item.product_name, + started_at=checkout_item.started_at, + stopped_at=checkout_item.stopped_at, + num_of_seats=checkout_item.num_of_seats, + ) async def checkout_licensed_item_for_wallet( @@ -33,7 +119,7 @@ async def checkout_licensed_item_for_wallet( licensed_item_id: LicensedItemID, num_of_seats: int, service_run_id: ServiceRunID, -) -> webserver_licensed_items_checkouts.LicensedItemCheckoutGet: +) -> LicensedItemCheckoutGet: # Check whether user has access to the wallet await get_wallet_by_user( app, @@ -58,7 +144,7 @@ async def checkout_licensed_item_for_wallet( ) ) - return webserver_licensed_items_checkouts.LicensedItemCheckoutGet( + return LicensedItemCheckoutGet.model_construct( licensed_item_checkout_id=licensed_item_get.licensed_item_checkout_id, licensed_item_id=licensed_item_get.licensed_item_id, wallet_id=licensed_item_get.wallet_id, @@ -78,7 +164,7 @@ async def release_licensed_item_for_wallet( user_id: UserID, # release args licensed_item_checkout_id: LicensedItemCheckoutID, -) -> webserver_licensed_items_checkouts.LicensedItemCheckoutGet: +) -> LicensedItemCheckoutGet: rpc_client = get_rabbitmq_rpc_client(app) checkout_item = await licensed_items_checkouts.get_licensed_item_checkout( @@ -103,7 +189,7 @@ async def release_licensed_item_for_wallet( ) ) - return webserver_licensed_items_checkouts.LicensedItemCheckoutGet( + return LicensedItemCheckoutGet.model_construct( licensed_item_checkout_id=licensed_item_get.licensed_item_checkout_id, licensed_item_id=licensed_item_get.licensed_item_id, wallet_id=licensed_item_get.wallet_id, diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py new file mode 100644 index 00000000000..3a1a18bbef3 --- /dev/null +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py @@ -0,0 +1,131 @@ +import logging + +from aiohttp import web +from models_library.api_schemas_webserver.licensed_items_checkouts import ( + LicensedItemCheckoutRestGet, + LicensedItemCheckoutRestGetPage, +) +from models_library.rest_ordering import OrderBy +from models_library.rest_pagination import Page +from models_library.rest_pagination_utils import paginate_data +from servicelib.aiohttp.requests_validation import ( + parse_request_path_parameters_as, + parse_request_query_parameters_as, +) +from servicelib.mimetype_constants import MIMETYPE_APPLICATION_JSON +from servicelib.rest_constants import RESPONSE_MODEL_POLICY + +from .._meta import API_VTAG as VTAG +from ..login.decorators import login_required +from ..security.decorators import permission_required +from ..utils_aiohttp import envelope_json_response +from ..wallets._handlers import WalletsPathParams +from . import _licensed_items_checkouts_api +from ._exceptions_handlers import handle_plugin_requests_exceptions +from ._licensed_items_checkouts_models import ( + LicensedItemCheckoutGet, + LicensedItemCheckoutGetPage, + LicensedItemCheckoutPathParams, +) +from ._models import LicensedItemsPurchasesListQueryParams, LicensedItemsRequestContext + +_logger = logging.getLogger(__name__) + + +routes = web.RouteTableDef() + + +@routes.get( + f"/{VTAG}/licensed-items-checkouts/{{licensed_item_checkout_id}}", + name="get_licensed_item_checkout", +) +@login_required +@permission_required("catalog/licensed-items.*") +@handle_plugin_requests_exceptions +async def get_licensed_item_checkout(request: web.Request): + req_ctx = LicensedItemsRequestContext.model_validate(request) + path_params = parse_request_path_parameters_as( + LicensedItemCheckoutPathParams, request + ) + + checkout_item: LicensedItemCheckoutGet = ( + await _licensed_items_checkouts_api.get_licensed_item_checkout( + app=request.app, + product_name=req_ctx.product_name, + user_id=req_ctx.user_id, + licensed_item_checkout_id=path_params.licensed_item_checkout_id, + ) + ) + + output = LicensedItemCheckoutRestGet.model_construct( + licensed_item_checkout_id=checkout_item.licensed_item_checkout_id, + licensed_item_id=checkout_item.licensed_item_id, + wallet_id=checkout_item.wallet_id, + user_id=checkout_item.user_id, + product_name=checkout_item.product_name, + started_at=checkout_item.started_at, + stopped_at=checkout_item.stopped_at, + num_of_seats=checkout_item.num_of_seats, + ) + + return envelope_json_response(output) + + +@routes.get( + f"/{VTAG}/wallets/{{wallet_id}}/licensed-items-checkouts", + name="list_licensed_item_checkouts_for_wallet", +) +@login_required +@permission_required("catalog/licensed-items.*") +@handle_plugin_requests_exceptions +async def list_licensed_item_checkouts_for_wallet(request: web.Request): + req_ctx = LicensedItemsRequestContext.model_validate(request) + path_params = parse_request_path_parameters_as(WalletsPathParams, request) + query_params: LicensedItemsPurchasesListQueryParams = ( + parse_request_query_parameters_as( + LicensedItemsPurchasesListQueryParams, request + ) + ) + + result: LicensedItemCheckoutGetPage = ( + await _licensed_items_checkouts_api.list_licensed_items_checkouts_for_wallet( + app=request.app, + product_name=req_ctx.product_name, + user_id=req_ctx.user_id, + wallet_id=path_params.wallet_id, + offset=query_params.offset, + limit=query_params.limit, + order_by=OrderBy.model_construct(**query_params.order_by.model_dump()), + ) + ) + + get_page = LicensedItemCheckoutRestGetPage( + total=result.total, + items=[ + LicensedItemCheckoutRestGet.model_construct( + licensed_item_checkout_id=checkout_item.licensed_item_checkout_id, + licensed_item_id=checkout_item.licensed_item_id, + wallet_id=checkout_item.wallet_id, + user_id=checkout_item.user_id, + product_name=checkout_item.product_name, + started_at=checkout_item.started_at, + stopped_at=checkout_item.stopped_at, + num_of_seats=checkout_item.num_of_seats, + ) + for checkout_item in result.items + ], + ) + + page = Page[LicensedItemCheckoutRestGetPage].model_validate( + paginate_data( + chunk=get_page.items, + request_url=request.url, + total=get_page.total, + limit=query_params.limit, + offset=query_params.offset, + ) + ) + return web.Response( + text=page.model_dump_json(**RESPONSE_MODEL_POLICY), + content_type=MIMETYPE_APPLICATION_JSON, + ) diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_models.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_models.py new file mode 100644 index 00000000000..43d6a290d6f --- /dev/null +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_models.py @@ -0,0 +1,56 @@ +from datetime import datetime +from typing import NamedTuple + +from models_library.basic_types import IDStr +from models_library.licensed_items import LicensedItemID +from models_library.products import ProductName +from models_library.resource_tracker_licensed_items_checkouts import ( + LicensedItemCheckoutID, +) +from models_library.rest_base import RequestParameters, StrictRequestParameters +from models_library.rest_ordering import ( + OrderBy, + OrderDirection, + create_ordering_query_model_class, +) +from models_library.rest_pagination import PageQueryParameters +from models_library.users import UserID +from models_library.wallets import WalletID +from pydantic import BaseModel, PositiveInt + + +class LicensedItemCheckoutGet(BaseModel): + licensed_item_checkout_id: LicensedItemCheckoutID + licensed_item_id: LicensedItemID + wallet_id: WalletID + user_id: UserID + product_name: ProductName + started_at: datetime + stopped_at: datetime | None + num_of_seats: int + + +class LicensedItemCheckoutGetPage(NamedTuple): + items: list[LicensedItemCheckoutGet] + total: PositiveInt + + +class LicensedItemCheckoutPathParams(StrictRequestParameters): + licensed_item_checkout_id: LicensedItemCheckoutID + + +_LicensedItemsCheckoutsListOrderQueryParams: type[ + RequestParameters +] = create_ordering_query_model_class( + ordering_fields={ + "started_at", + }, + default=OrderBy(field=IDStr("started_at"), direction=OrderDirection.DESC), +) + + +class LicensedItemsCheckoutsListQueryParams( + PageQueryParameters, + _LicensedItemsCheckoutsListOrderQueryParams, # type: ignore[misc, valid-type] +): + ... diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_api.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_api.py index b42d593bed1..2cfa6355f83 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_api.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_api.py @@ -75,6 +75,7 @@ async def list_licensed_items_purchases( async def get_licensed_item_purchase( app: web.Application, + *, product_name: ProductName, user_id: UserID, licensed_item_purchase_id: LicensedItemPurchaseID, diff --git a/services/web/server/src/simcore_service_webserver/licenses/_rpc.py b/services/web/server/src/simcore_service_webserver/licenses/_rpc.py index 8be794e4016..3ecd72e65d6 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_rpc.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_rpc.py @@ -2,7 +2,7 @@ from models_library.api_schemas_webserver import WEBSERVER_RPC_NAMESPACE from models_library.api_schemas_webserver.licensed_items import LicensedItemGetPage from models_library.api_schemas_webserver.licensed_items_checkouts import ( - LicensedItemCheckoutGet, + LicensedItemCheckoutRpcGet, ) from models_library.basic_types import IDStr from models_library.licensed_items import LicensedItemID @@ -20,7 +20,7 @@ ) from ..rabbitmq import get_rabbitmq_rpc_server -from . import _licensed_checkouts_api, _licensed_items_api +from . import _licensed_items_api, _licensed_items_checkouts_api router = RPCRouter() @@ -68,15 +68,27 @@ async def checkout_licensed_item_for_wallet( licensed_item_id: LicensedItemID, num_of_seats: int, service_run_id: ServiceRunID, -) -> LicensedItemCheckoutGet: - return await _licensed_checkouts_api.checkout_licensed_item_for_wallet( - app, - licensed_item_id=licensed_item_id, - wallet_id=wallet_id, - product_name=product_name, - num_of_seats=num_of_seats, - service_run_id=service_run_id, - user_id=user_id, +) -> LicensedItemCheckoutRpcGet: + licensed_item_get = ( + await _licensed_items_checkouts_api.checkout_licensed_item_for_wallet( + app, + licensed_item_id=licensed_item_id, + wallet_id=wallet_id, + product_name=product_name, + num_of_seats=num_of_seats, + service_run_id=service_run_id, + user_id=user_id, + ) + ) + return LicensedItemCheckoutRpcGet.model_construct( + licensed_item_checkout_id=licensed_item_get.licensed_item_checkout_id, + licensed_item_id=licensed_item_get.licensed_item_id, + wallet_id=licensed_item_get.wallet_id, + user_id=licensed_item_get.user_id, + product_name=licensed_item_get.product_name, + started_at=licensed_item_get.started_at, + stopped_at=licensed_item_get.stopped_at, + num_of_seats=licensed_item_get.num_of_seats, ) @@ -87,12 +99,24 @@ async def release_licensed_item_for_wallet( product_name: ProductName, user_id: UserID, licensed_item_checkout_id: LicensedItemCheckoutID, -) -> LicensedItemCheckoutGet: - return await _licensed_checkouts_api.release_licensed_item_for_wallet( - app, - product_name=product_name, - user_id=user_id, - licensed_item_checkout_id=licensed_item_checkout_id, +) -> LicensedItemCheckoutRpcGet: + licensed_item_get = ( + await _licensed_items_checkouts_api.release_licensed_item_for_wallet( + app, + product_name=product_name, + user_id=user_id, + licensed_item_checkout_id=licensed_item_checkout_id, + ) + ) + return LicensedItemCheckoutRpcGet.model_construct( + licensed_item_checkout_id=licensed_item_get.licensed_item_checkout_id, + licensed_item_id=licensed_item_get.licensed_item_id, + wallet_id=licensed_item_get.wallet_id, + user_id=licensed_item_get.user_id, + product_name=licensed_item_get.product_name, + started_at=licensed_item_get.started_at, + stopped_at=licensed_item_get.stopped_at, + num_of_seats=licensed_item_get.num_of_seats, ) diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py new file mode 100644 index 00000000000..b6e60997e20 --- /dev/null +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py @@ -0,0 +1,84 @@ +# pylint: disable=redefined-outer-name +# pylint: disable=unused-argument +# pylint: disable=unused-variable +# pylint: disable=too-many-arguments +# pylint: disable=too-many-statements +from http import HTTPStatus + +import pytest +from aiohttp.test_utils import TestClient +from models_library.api_schemas_resource_usage_tracker import ( + licensed_items_checkouts as rut_licensed_items_checkouts, +) +from models_library.api_schemas_webserver.licensed_items_purchases import ( + LicensedItemPurchaseGet, +) +from pytest_mock.plugin import MockerFixture +from pytest_simcore.helpers.assert_checks import assert_status +from pytest_simcore.helpers.webserver_login import UserInfoDict +from servicelib.aiohttp import status +from simcore_service_webserver.db.models import UserRole + +_LICENSED_ITEM_CHECKOUT_GET = ( + rut_licensed_items_checkouts.LicensedItemCheckoutGet.model_validate({}) +) + +_LICENSED_ITEM_CHECKOUT_PAGE = rut_licensed_items_checkouts.LicensedItemsCheckoutsPage( + items=[_LICENSED_ITEM_CHECKOUT_GET], + total=1, +) + + +@pytest.fixture +def mock_get_licensed_items_purchases_page(mocker: MockerFixture) -> tuple: + return mocker.patch( + "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts_api.get_licensed_items_purchases_page", + spec=True, + return_value=_LICENSED_ITEM_CHECKOUT_PAGE, + ) + + +@pytest.fixture +def mock_get_licensed_item_purchase(mocker: MockerFixture) -> tuple: + return mocker.patch( + "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_purchases.get_licensed_item_purchase", + spec=True, + return_value=_LICENSED_ITEM_CHECKOUT_GET, + ) + + +@pytest.fixture +def mock_get_wallet_by_user(mocker: MockerFixture) -> tuple: + return mocker.patch( + "simcore_service_webserver.licenses._licensed_items_checkouts_api.get_wallet_by_user", + spec=True, + ) + + +@pytest.mark.parametrize("user_role,expected", [(UserRole.USER, status.HTTP_200_OK)]) +async def test_licensed_items_checkouts_handlers( + client: TestClient, + logged_user: UserInfoDict, + expected: HTTPStatus, + mock_get_licensed_items_purchases_page: MockerFixture, + mock_get_licensed_item_purchase: MockerFixture, + mock_get_wallet_by_user: MockerFixture, +): + assert client.app + + # list + url = client.app.router["list_wallet_licensed_items_purchases"].url_for( + wallet_id="1" + ) + resp = await client.get(f"{url}") + data, _ = await assert_status(resp, status.HTTP_200_OK) + assert len(data) == 1 + assert LicensedItemPurchaseGet(**data[0]) + + # get + url = client.app.router["get_licensed_item_purchase"].url_for( + licensed_item_purchase_id=f"{_LICENSED_ITEM_PURCHASE_PAGE.items[0].licensed_item_purchase_id}" + ) + resp = await client.get(f"{url}") + data, _ = await assert_status(resp, status.HTTP_200_OK) + assert LicensedItemPurchaseGet(**data) diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_handlers.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_handlers.py index ce0fddeca19..23e10209f27 100644 --- a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_handlers.py +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_handlers.py @@ -73,7 +73,7 @@ def mock_get_wallet_by_user(mocker: MockerFixture) -> tuple: @pytest.mark.parametrize("user_role,expected", [(UserRole.USER, status.HTTP_200_OK)]) -async def test_licensed_items_db_crud( +async def test_licensed_items_purchaches_handlers( client: TestClient, logged_user: UserInfoDict, expected: HTTPStatus, diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licenses_rpc.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licenses_rpc.py index 6888711b2da..7bbfae01c90 100644 --- a/services/web/server/tests/unit/with_dbs/04/licenses/test_licenses_rpc.py +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licenses_rpc.py @@ -73,7 +73,7 @@ async def rpc_client( @pytest.fixture def mock_get_wallet_by_user(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_checkouts_api.get_wallet_by_user", + "simcore_service_webserver.licenses._licensed_items_checkouts_api.get_wallet_by_user", spec=True, ) @@ -86,7 +86,7 @@ def mock_get_wallet_by_user(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_checkout_licensed_item(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_checkouts_api.licensed_items_checkouts.checkout_licensed_item", + "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.checkout_licensed_item", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_GET, ) @@ -95,7 +95,7 @@ def mock_checkout_licensed_item(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_get_licensed_item_checkout(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_checkouts_api.licensed_items_checkouts.get_licensed_item_checkout", + "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.get_licensed_item_checkout", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_GET, ) @@ -104,7 +104,7 @@ def mock_get_licensed_item_checkout(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_release_licensed_item(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_checkouts_api.licensed_items_checkouts.release_licensed_item", + "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.release_licensed_item", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_GET, ) From 53364d0d6e2641481357fe40d0c3b62a3a616f2b Mon Sep 17 00:00:00 2001 From: matusdrobuliak66 Date: Fri, 20 Dec 2024 15:15:11 +0100 Subject: [PATCH 5/7] list/get checkouts --- .../_licensed_items_checkouts_handlers.py | 2 +- .../licenses/plugin.py | 8 +++- .../test_licensed_items_checkouts_handlers.py | 37 ++++++++++--------- 3 files changed, 27 insertions(+), 20 deletions(-) diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py index 3a1a18bbef3..441a08fdf41 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py @@ -116,7 +116,7 @@ async def list_licensed_item_checkouts_for_wallet(request: web.Request): ], ) - page = Page[LicensedItemCheckoutRestGetPage].model_validate( + page = Page[LicensedItemCheckoutRestGet].model_validate( paginate_data( chunk=get_page.items, request_url=request.url, diff --git a/services/web/server/src/simcore_service_webserver/licenses/plugin.py b/services/web/server/src/simcore_service_webserver/licenses/plugin.py index 137c7b2d1dc..2e2a04517a5 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/plugin.py +++ b/services/web/server/src/simcore_service_webserver/licenses/plugin.py @@ -8,7 +8,12 @@ from servicelib.aiohttp.application_setup import ModuleCategory, app_module_setup from ..rabbitmq import setup_rabbitmq -from . import _licensed_items_handlers, _licensed_items_purchases_handlers, _rpc +from . import ( + _licensed_items_checkouts_handlers, + _licensed_items_handlers, + _licensed_items_purchases_handlers, + _rpc, +) _logger = logging.getLogger(__name__) @@ -26,6 +31,7 @@ def setup_licenses(app: web.Application): # routes app.router.add_routes(_licensed_items_handlers.routes) app.router.add_routes(_licensed_items_purchases_handlers.routes) + app.router.add_routes(_licensed_items_checkouts_handlers.routes) setup_rabbitmq(app) if app[APP_SETTINGS_KEY].WEBSERVER_RABBITMQ: diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py index b6e60997e20..f1573def540 100644 --- a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py @@ -7,11 +7,12 @@ import pytest from aiohttp.test_utils import TestClient -from models_library.api_schemas_resource_usage_tracker import ( - licensed_items_checkouts as rut_licensed_items_checkouts, +from models_library.api_schemas_resource_usage_tracker.licensed_items_checkouts import ( + LicensedItemCheckoutGet, + LicensedItemsCheckoutsPage, ) -from models_library.api_schemas_webserver.licensed_items_purchases import ( - LicensedItemPurchaseGet, +from models_library.api_schemas_webserver.licensed_items_checkouts import ( + LicensedItemCheckoutRestGet, ) from pytest_mock.plugin import MockerFixture from pytest_simcore.helpers.assert_checks import assert_status @@ -19,29 +20,29 @@ from servicelib.aiohttp import status from simcore_service_webserver.db.models import UserRole -_LICENSED_ITEM_CHECKOUT_GET = ( - rut_licensed_items_checkouts.LicensedItemCheckoutGet.model_validate({}) +_LICENSED_ITEM_CHECKOUT_GET = LicensedItemCheckoutGet.model_validate( + LicensedItemCheckoutGet.model_config["json_schema_extra"]["examples"][0] ) -_LICENSED_ITEM_CHECKOUT_PAGE = rut_licensed_items_checkouts.LicensedItemsCheckoutsPage( +_LICENSED_ITEM_CHECKOUT_PAGE = LicensedItemsCheckoutsPage( items=[_LICENSED_ITEM_CHECKOUT_GET], total=1, ) @pytest.fixture -def mock_get_licensed_items_purchases_page(mocker: MockerFixture) -> tuple: +def mock_get_licensed_items_checkouts_page(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts_api.get_licensed_items_purchases_page", + "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.get_licensed_items_checkouts_page", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_PAGE, ) @pytest.fixture -def mock_get_licensed_item_purchase(mocker: MockerFixture) -> tuple: +def mock_get_licensed_item_checkout(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_purchases.get_licensed_item_purchase", + "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.get_licensed_item_checkout", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_GET, ) @@ -60,25 +61,25 @@ async def test_licensed_items_checkouts_handlers( client: TestClient, logged_user: UserInfoDict, expected: HTTPStatus, - mock_get_licensed_items_purchases_page: MockerFixture, - mock_get_licensed_item_purchase: MockerFixture, + mock_get_licensed_items_checkouts_page: MockerFixture, + mock_get_licensed_item_checkout: MockerFixture, mock_get_wallet_by_user: MockerFixture, ): assert client.app # list - url = client.app.router["list_wallet_licensed_items_purchases"].url_for( + url = client.app.router["list_licensed_item_checkouts_for_wallet"].url_for( wallet_id="1" ) resp = await client.get(f"{url}") data, _ = await assert_status(resp, status.HTTP_200_OK) assert len(data) == 1 - assert LicensedItemPurchaseGet(**data[0]) + assert LicensedItemCheckoutRestGet(**data[0]) # get - url = client.app.router["get_licensed_item_purchase"].url_for( - licensed_item_purchase_id=f"{_LICENSED_ITEM_PURCHASE_PAGE.items[0].licensed_item_purchase_id}" + url = client.app.router["get_licensed_item_checkout"].url_for( + licensed_item_checkout_id=f"{_LICENSED_ITEM_CHECKOUT_PAGE.items[0].licensed_item_checkout_id}" ) resp = await client.get(f"{url}") data, _ = await assert_status(resp, status.HTTP_200_OK) - assert LicensedItemPurchaseGet(**data) + assert LicensedItemCheckoutRestGet(**data) From 381dcc3b23f5fb32513950ecbe240adfdacdaf37 Mon Sep 17 00:00:00 2001 From: matusdrobuliak66 Date: Fri, 20 Dec 2024 15:44:10 +0100 Subject: [PATCH 6/7] openapi specs --- api/specs/web-server/openapi.py | 1 + .../api/v0/openapi.yaml | 115 ++++++++++++++++++ 2 files changed, 116 insertions(+) diff --git a/api/specs/web-server/openapi.py b/api/specs/web-server/openapi.py index 6c3bc639fb4..cfcaf183591 100644 --- a/api/specs/web-server/openapi.py +++ b/api/specs/web-server/openapi.py @@ -38,6 +38,7 @@ "_long_running_tasks", "_licensed_items", "_licensed_items_purchases", + "_licensed_items_checkouts", "_metamodeling", "_nih_sparc", "_nih_sparc_redirections", diff --git a/services/web/server/src/simcore_service_webserver/api/v0/openapi.yaml b/services/web/server/src/simcore_service_webserver/api/v0/openapi.yaml index 5e20d12a5ab..3f547af4b79 100644 --- a/services/web/server/src/simcore_service_webserver/api/v0/openapi.yaml +++ b/services/web/server/src/simcore_service_webserver/api/v0/openapi.yaml @@ -3252,6 +3252,121 @@ paths: schema: $ref: '#/components/schemas/EnvelopedError' description: Bad Request + /v0/wallets/{wallet_id}/licensed-items-checkouts: + get: + tags: + - licenses + - wallets + summary: List Licensed Item Checkouts For Wallet + operationId: list_licensed_item_checkouts_for_wallet + parameters: + - name: wallet_id + in: path + required: true + schema: + type: integer + exclusiveMinimum: true + title: Wallet Id + minimum: 0 + - name: order_by + in: query + required: false + schema: + type: string + contentMediaType: application/json + contentSchema: {} + default: '{"field":"started_at","direction":"desc"}' + title: Order By + - name: limit + in: query + required: false + schema: + type: integer + default: 20 + title: Limit + - name: offset + in: query + required: false + schema: + type: integer + default: 0 + title: Offset + responses: + '200': + description: Successful Response + content: + application/json: + schema: + $ref: '#/components/schemas/Page_LicensedItemPurchaseGet_' + '404': + content: + application/json: + schema: + $ref: '#/components/schemas/EnvelopedError' + description: Not Found + '403': + content: + application/json: + schema: + $ref: '#/components/schemas/EnvelopedError' + description: Forbidden + '402': + content: + application/json: + schema: + $ref: '#/components/schemas/EnvelopedError' + description: Payment Required + '400': + content: + application/json: + schema: + $ref: '#/components/schemas/EnvelopedError' + description: Bad Request + /v0/licensed-items-checkouts/{licensed_item_checkout_id}: + get: + tags: + - licenses + summary: Get Licensed Item Checkout + operationId: get_licensed_item_checkout + parameters: + - name: licensed_item_checkout_id + in: path + required: true + schema: + type: string + format: uuid + title: Licensed Item Checkout Id + responses: + '200': + description: Successful Response + content: + application/json: + schema: + $ref: '#/components/schemas/Envelope_LicensedItemPurchaseGet_' + '404': + content: + application/json: + schema: + $ref: '#/components/schemas/EnvelopedError' + description: Not Found + '403': + content: + application/json: + schema: + $ref: '#/components/schemas/EnvelopedError' + description: Forbidden + '402': + content: + application/json: + schema: + $ref: '#/components/schemas/EnvelopedError' + description: Payment Required + '400': + content: + application/json: + schema: + $ref: '#/components/schemas/EnvelopedError' + description: Bad Request /v0/projects/{project_uuid}/checkpoint/{ref_id}/iterations: get: tags: From 75e4f01420ea03a1f30b07c2a5aae2625d0e0b5e Mon Sep 17 00:00:00 2001 From: matusdrobuliak66 Date: Mon, 6 Jan 2025 13:13:08 +0100 Subject: [PATCH 7/7] renaming based on new standars --- ...s.py => _licensed_items_checkouts_rest.py} | 22 +++++++++---------- ...y => _licensed_items_checkouts_service.py} | 0 ...s.py => _licensed_items_purchases_rest.py} | 6 ++--- ...y => _licensed_items_purchases_service.py} | 0 ...ms_db.py => _licensed_items_repository.py} | 0 ...ms_handlers.py => _licensed_items_rest.py} | 16 ++++++++------ ...tems_api.py => _licensed_items_service.py} | 6 ++--- .../licenses/_rpc.py | 8 +++---- .../licenses/plugin.py | 12 +++++----- .../licenses/{api.py => service.py} | 0 ... => test_licensed_items_checkouts_rest.py} | 6 ++--- ... => test_licensed_items_purchases_rest.py} | 6 ++--- ...b.py => test_licensed_items_repository.py} | 18 +++++++-------- ...andlers.py => test_licensed_items_rest.py} | 12 +++++----- .../with_dbs/04/licenses/test_licenses_rpc.py | 12 +++++----- 15 files changed, 62 insertions(+), 62 deletions(-) rename services/web/server/src/simcore_service_webserver/licenses/{_licensed_items_checkouts_handlers.py => _licensed_items_checkouts_rest.py} (87%) rename services/web/server/src/simcore_service_webserver/licenses/{_licensed_items_checkouts_api.py => _licensed_items_checkouts_service.py} (100%) rename services/web/server/src/simcore_service_webserver/licenses/{_licensed_items_purchases_handlers.py => _licensed_items_purchases_rest.py} (94%) rename services/web/server/src/simcore_service_webserver/licenses/{_licensed_items_purchases_api.py => _licensed_items_purchases_service.py} (100%) rename services/web/server/src/simcore_service_webserver/licenses/{_licensed_items_db.py => _licensed_items_repository.py} (100%) rename services/web/server/src/simcore_service_webserver/licenses/{_licensed_items_handlers.py => _licensed_items_rest.py} (89%) rename services/web/server/src/simcore_service_webserver/licenses/{_licensed_items_api.py => _licensed_items_service.py} (96%) rename services/web/server/src/simcore_service_webserver/licenses/{api.py => service.py} (100%) rename services/web/server/tests/unit/with_dbs/04/licenses/{test_licensed_items_checkouts_handlers.py => test_licensed_items_checkouts_rest.py} (94%) rename services/web/server/tests/unit/with_dbs/04/licenses/{test_licensed_items_purchases_handlers.py => test_licensed_items_purchases_rest.py} (95%) rename services/web/server/tests/unit/with_dbs/04/licenses/{test_licensed_items_db.py => test_licensed_items_repository.py} (84%) rename services/web/server/tests/unit/with_dbs/04/licenses/{test_licensed_items_handlers.py => test_licensed_items_rest.py} (88%) diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_rest.py similarity index 87% rename from services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py rename to services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_rest.py index 441a08fdf41..1a9c7285d0a 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_handlers.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_rest.py @@ -20,7 +20,7 @@ from ..security.decorators import permission_required from ..utils_aiohttp import envelope_json_response from ..wallets._handlers import WalletsPathParams -from . import _licensed_items_checkouts_api +from . import _licensed_items_checkouts_service from ._exceptions_handlers import handle_plugin_requests_exceptions from ._licensed_items_checkouts_models import ( LicensedItemCheckoutGet, @@ -49,7 +49,7 @@ async def get_licensed_item_checkout(request: web.Request): ) checkout_item: LicensedItemCheckoutGet = ( - await _licensed_items_checkouts_api.get_licensed_item_checkout( + await _licensed_items_checkouts_service.get_licensed_item_checkout( app=request.app, product_name=req_ctx.product_name, user_id=req_ctx.user_id, @@ -87,16 +87,14 @@ async def list_licensed_item_checkouts_for_wallet(request: web.Request): ) ) - result: LicensedItemCheckoutGetPage = ( - await _licensed_items_checkouts_api.list_licensed_items_checkouts_for_wallet( - app=request.app, - product_name=req_ctx.product_name, - user_id=req_ctx.user_id, - wallet_id=path_params.wallet_id, - offset=query_params.offset, - limit=query_params.limit, - order_by=OrderBy.model_construct(**query_params.order_by.model_dump()), - ) + result: LicensedItemCheckoutGetPage = await _licensed_items_checkouts_service.list_licensed_items_checkouts_for_wallet( + app=request.app, + product_name=req_ctx.product_name, + user_id=req_ctx.user_id, + wallet_id=path_params.wallet_id, + offset=query_params.offset, + limit=query_params.limit, + order_by=OrderBy.model_construct(**query_params.order_by.model_dump()), ) get_page = LicensedItemCheckoutRestGetPage( diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_api.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_service.py similarity index 100% rename from services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_api.py rename to services/web/server/src/simcore_service_webserver/licenses/_licensed_items_checkouts_service.py diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_handlers.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_rest.py similarity index 94% rename from services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_handlers.py rename to services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_rest.py index 95f48ebbd0e..5ae0738ebe1 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_handlers.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_rest.py @@ -20,7 +20,7 @@ from ..security.decorators import permission_required from ..utils_aiohttp import envelope_json_response from ..wallets._handlers import WalletsPathParams -from . import _licensed_items_purchases_api +from . import _licensed_items_purchases_service from ._exceptions_handlers import handle_plugin_requests_exceptions from ._models import ( LicensedItemsPurchasesListQueryParams, @@ -47,7 +47,7 @@ async def get_licensed_item_purchase(request: web.Request): ) licensed_item_purchase_get: LicensedItemPurchaseGet = ( - await _licensed_items_purchases_api.get_licensed_item_purchase( + await _licensed_items_purchases_service.get_licensed_item_purchase( app=request.app, product_name=req_ctx.product_name, user_id=req_ctx.user_id, @@ -75,7 +75,7 @@ async def list_wallet_licensed_items_purchases(request: web.Request): ) licensed_item_purchase_get_page: LicensedItemPurchaseGetPage = ( - await _licensed_items_purchases_api.list_licensed_items_purchases( + await _licensed_items_purchases_service.list_licensed_items_purchases( app=request.app, product_name=req_ctx.product_name, user_id=req_ctx.user_id, diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_api.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_service.py similarity index 100% rename from services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_api.py rename to services/web/server/src/simcore_service_webserver/licenses/_licensed_items_purchases_service.py diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_db.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_repository.py similarity index 100% rename from services/web/server/src/simcore_service_webserver/licenses/_licensed_items_db.py rename to services/web/server/src/simcore_service_webserver/licenses/_licensed_items_repository.py diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_handlers.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_rest.py similarity index 89% rename from services/web/server/src/simcore_service_webserver/licenses/_licensed_items_handlers.py rename to services/web/server/src/simcore_service_webserver/licenses/_licensed_items_rest.py index 355d9658ebb..4f0a936c041 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_handlers.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_rest.py @@ -21,7 +21,7 @@ from ..login.decorators import login_required from ..security.decorators import permission_required from ..utils_aiohttp import envelope_json_response -from . import _licensed_items_api +from . import _licensed_items_service from ._exceptions_handlers import handle_plugin_requests_exceptions from ._models import ( LicensedItemsBodyParams, @@ -47,7 +47,7 @@ async def list_licensed_items(request: web.Request): ) licensed_item_get_page: LicensedItemGetPage = ( - await _licensed_items_api.list_licensed_items( + await _licensed_items_service.list_licensed_items( app=request.app, product_name=req_ctx.product_name, offset=query_params.offset, @@ -81,10 +81,12 @@ async def get_licensed_item(request: web.Request): req_ctx = LicensedItemsRequestContext.model_validate(request) path_params = parse_request_path_parameters_as(LicensedItemsPathParams, request) - licensed_item_get: LicensedItemGet = await _licensed_items_api.get_licensed_item( - app=request.app, - licensed_item_id=path_params.licensed_item_id, - product_name=req_ctx.product_name, + licensed_item_get: LicensedItemGet = ( + await _licensed_items_service.get_licensed_item( + app=request.app, + licensed_item_id=path_params.licensed_item_id, + product_name=req_ctx.product_name, + ) ) return envelope_json_response(licensed_item_get) @@ -102,7 +104,7 @@ async def purchase_licensed_item(request: web.Request): path_params = parse_request_path_parameters_as(LicensedItemsPathParams, request) body_params = await parse_request_body_as(LicensedItemsBodyParams, request) - await _licensed_items_api.purchase_licensed_item( + await _licensed_items_service.purchase_licensed_item( app=request.app, user_id=req_ctx.user_id, licensed_item_id=path_params.licensed_item_id, diff --git a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_api.py b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_service.py similarity index 96% rename from services/web/server/src/simcore_service_webserver/licenses/_licensed_items_api.py rename to services/web/server/src/simcore_service_webserver/licenses/_licensed_items_service.py index 1f839ae31fa..374da33bbbe 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_api.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_licensed_items_service.py @@ -25,7 +25,7 @@ from ..users.api import get_user from ..wallets.api import get_wallet_with_available_credits_by_user_and_wallet from ..wallets.errors import WalletNotEnoughCreditsError -from . import _licensed_items_db +from . import _licensed_items_repository from ._models import LicensedItemsBodyParams from .errors import LicensedItemPricingPlanMatchError @@ -39,7 +39,7 @@ async def get_licensed_item( product_name: ProductName, ) -> LicensedItemGet: - licensed_item_db = await _licensed_items_db.get( + licensed_item_db = await _licensed_items_repository.get( app, licensed_item_id=licensed_item_id, product_name=product_name ) return LicensedItemGet( @@ -61,7 +61,7 @@ async def list_licensed_items( limit: int, order_by: OrderBy, ) -> LicensedItemGetPage: - total_count, licensed_item_db_list = await _licensed_items_db.list_( + total_count, licensed_item_db_list = await _licensed_items_repository.list_( app, product_name=product_name, offset=offset, limit=limit, order_by=order_by ) return LicensedItemGetPage( diff --git a/services/web/server/src/simcore_service_webserver/licenses/_rpc.py b/services/web/server/src/simcore_service_webserver/licenses/_rpc.py index 3ecd72e65d6..261eb51c3aa 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/_rpc.py +++ b/services/web/server/src/simcore_service_webserver/licenses/_rpc.py @@ -20,7 +20,7 @@ ) from ..rabbitmq import get_rabbitmq_rpc_server -from . import _licensed_items_api, _licensed_items_checkouts_api +from . import _licensed_items_checkouts_service, _licensed_items_service router = RPCRouter() @@ -34,7 +34,7 @@ async def get_licensed_items( limit: int, ) -> LicensedItemGetPage: licensed_item_get_page: LicensedItemGetPage = ( - await _licensed_items_api.list_licensed_items( + await _licensed_items_service.list_licensed_items( app=app, product_name=product_name, offset=offset, @@ -70,7 +70,7 @@ async def checkout_licensed_item_for_wallet( service_run_id: ServiceRunID, ) -> LicensedItemCheckoutRpcGet: licensed_item_get = ( - await _licensed_items_checkouts_api.checkout_licensed_item_for_wallet( + await _licensed_items_checkouts_service.checkout_licensed_item_for_wallet( app, licensed_item_id=licensed_item_id, wallet_id=wallet_id, @@ -101,7 +101,7 @@ async def release_licensed_item_for_wallet( licensed_item_checkout_id: LicensedItemCheckoutID, ) -> LicensedItemCheckoutRpcGet: licensed_item_get = ( - await _licensed_items_checkouts_api.release_licensed_item_for_wallet( + await _licensed_items_checkouts_service.release_licensed_item_for_wallet( app, product_name=product_name, user_id=user_id, diff --git a/services/web/server/src/simcore_service_webserver/licenses/plugin.py b/services/web/server/src/simcore_service_webserver/licenses/plugin.py index 2e2a04517a5..72af99badeb 100644 --- a/services/web/server/src/simcore_service_webserver/licenses/plugin.py +++ b/services/web/server/src/simcore_service_webserver/licenses/plugin.py @@ -9,9 +9,9 @@ from ..rabbitmq import setup_rabbitmq from . import ( - _licensed_items_checkouts_handlers, - _licensed_items_handlers, - _licensed_items_purchases_handlers, + _licensed_items_checkouts_rest, + _licensed_items_purchases_rest, + _licensed_items_rest, _rpc, ) @@ -29,9 +29,9 @@ def setup_licenses(app: web.Application): assert app[APP_SETTINGS_KEY].WEBSERVER_LICENSES # nosec # routes - app.router.add_routes(_licensed_items_handlers.routes) - app.router.add_routes(_licensed_items_purchases_handlers.routes) - app.router.add_routes(_licensed_items_checkouts_handlers.routes) + app.router.add_routes(_licensed_items_rest.routes) + app.router.add_routes(_licensed_items_purchases_rest.routes) + app.router.add_routes(_licensed_items_checkouts_rest.routes) setup_rabbitmq(app) if app[APP_SETTINGS_KEY].WEBSERVER_RABBITMQ: diff --git a/services/web/server/src/simcore_service_webserver/licenses/api.py b/services/web/server/src/simcore_service_webserver/licenses/service.py similarity index 100% rename from services/web/server/src/simcore_service_webserver/licenses/api.py rename to services/web/server/src/simcore_service_webserver/licenses/service.py diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_rest.py similarity index 94% rename from services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py rename to services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_rest.py index f1573def540..1a6a81e76f4 100644 --- a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_handlers.py +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_checkouts_rest.py @@ -33,7 +33,7 @@ @pytest.fixture def mock_get_licensed_items_checkouts_page(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.get_licensed_items_checkouts_page", + "simcore_service_webserver.licenses._licensed_items_checkouts_service.licensed_items_checkouts.get_licensed_items_checkouts_page", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_PAGE, ) @@ -42,7 +42,7 @@ def mock_get_licensed_items_checkouts_page(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_get_licensed_item_checkout(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.get_licensed_item_checkout", + "simcore_service_webserver.licenses._licensed_items_checkouts_service.licensed_items_checkouts.get_licensed_item_checkout", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_GET, ) @@ -51,7 +51,7 @@ def mock_get_licensed_item_checkout(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_get_wallet_by_user(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.get_wallet_by_user", + "simcore_service_webserver.licenses._licensed_items_checkouts_service.get_wallet_by_user", spec=True, ) diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_handlers.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_rest.py similarity index 95% rename from services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_handlers.py rename to services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_rest.py index 23e10209f27..ee3656d2c1c 100644 --- a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_handlers.py +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_purchases_rest.py @@ -49,7 +49,7 @@ @pytest.fixture def mock_get_licensed_items_purchases_page(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_purchases_api.licensed_items_purchases.get_licensed_items_purchases_page", + "simcore_service_webserver.licenses._licensed_items_purchases_service.licensed_items_purchases.get_licensed_items_purchases_page", spec=True, return_value=_LICENSED_ITEM_PURCHASE_PAGE, ) @@ -58,7 +58,7 @@ def mock_get_licensed_items_purchases_page(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_get_licensed_item_purchase(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_purchases_api.licensed_items_purchases.get_licensed_item_purchase", + "simcore_service_webserver.licenses._licensed_items_purchases_service.licensed_items_purchases.get_licensed_item_purchase", spec=True, return_value=_LICENSED_ITEM_PURCHASE_GET, ) @@ -67,7 +67,7 @@ def mock_get_licensed_item_purchase(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_get_wallet_by_user(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_purchases_api.get_wallet_by_user", + "simcore_service_webserver.licenses._licensed_items_purchases_service.get_wallet_by_user", spec=True, ) diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_db.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_repository.py similarity index 84% rename from services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_db.py rename to services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_repository.py index 910e1bdf3f4..dfe04e2e0d3 100644 --- a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_db.py +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_repository.py @@ -16,7 +16,7 @@ from pytest_simcore.helpers.webserver_login import UserInfoDict from servicelib.aiohttp import status from simcore_service_webserver.db.models import UserRole -from simcore_service_webserver.licenses import _licensed_items_db +from simcore_service_webserver.licenses import _licensed_items_repository from simcore_service_webserver.licenses.errors import LicensedItemNotFoundError from simcore_service_webserver.projects.models import ProjectDict @@ -32,7 +32,7 @@ async def test_licensed_items_db_crud( ): assert client.app - output: tuple[int, list[LicensedItemDB]] = await _licensed_items_db.list_( + output: tuple[int, list[LicensedItemDB]] = await _licensed_items_repository.list_( client.app, product_name=osparc_product_name, offset=0, @@ -41,7 +41,7 @@ async def test_licensed_items_db_crud( ) assert output[0] == 0 - licensed_item_db = await _licensed_items_db.create( + licensed_item_db = await _licensed_items_repository.create( client.app, product_name=osparc_product_name, name="Model A", @@ -50,7 +50,7 @@ async def test_licensed_items_db_crud( ) _licensed_item_id = licensed_item_db.licensed_item_id - output: tuple[int, list[LicensedItemDB]] = await _licensed_items_db.list_( + output: tuple[int, list[LicensedItemDB]] = await _licensed_items_repository.list_( client.app, product_name=osparc_product_name, offset=0, @@ -59,35 +59,35 @@ async def test_licensed_items_db_crud( ) assert output[0] == 1 - licensed_item_db = await _licensed_items_db.get( + licensed_item_db = await _licensed_items_repository.get( client.app, licensed_item_id=_licensed_item_id, product_name=osparc_product_name, ) assert licensed_item_db.name == "Model A" - await _licensed_items_db.update( + await _licensed_items_repository.update( client.app, licensed_item_id=_licensed_item_id, product_name=osparc_product_name, updates=LicensedItemUpdateDB(name="Model B"), ) - licensed_item_db = await _licensed_items_db.get( + licensed_item_db = await _licensed_items_repository.get( client.app, licensed_item_id=_licensed_item_id, product_name=osparc_product_name, ) assert licensed_item_db.name == "Model B" - licensed_item_db = await _licensed_items_db.delete( + licensed_item_db = await _licensed_items_repository.delete( client.app, licensed_item_id=_licensed_item_id, product_name=osparc_product_name, ) with pytest.raises(LicensedItemNotFoundError): - await _licensed_items_db.get( + await _licensed_items_repository.get( client.app, licensed_item_id=_licensed_item_id, product_name=osparc_product_name, diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_handlers.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_rest.py similarity index 88% rename from services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_handlers.py rename to services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_rest.py index b1fee67dafa..67c36f2581b 100644 --- a/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_handlers.py +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licensed_items_rest.py @@ -18,7 +18,7 @@ from pytest_simcore.helpers.webserver_login import UserInfoDict from servicelib.aiohttp import status from simcore_service_webserver.db.models import UserRole -from simcore_service_webserver.licenses import _licensed_items_db +from simcore_service_webserver.licenses import _licensed_items_repository from simcore_service_webserver.projects.models import ProjectDict @@ -39,7 +39,7 @@ async def test_licensed_items_listing( data, _ = await assert_status(resp, status.HTTP_200_OK) assert data == [] - licensed_item_db = await _licensed_items_db.create( + licensed_item_db = await _licensed_items_repository.create( client.app, product_name=osparc_product_name, name="Model A", @@ -67,7 +67,7 @@ async def test_licensed_items_listing( @pytest.fixture def mock_licensed_items_purchase_functions(mocker: MockerFixture) -> tuple: mock_wallet_credits = mocker.patch( - "simcore_service_webserver.licenses._licensed_items_api.get_wallet_with_available_credits_by_user_and_wallet", + "simcore_service_webserver.licenses._licensed_items_service.get_wallet_with_available_credits_by_user_and_wallet", spec=True, return_value=WalletGetWithAvailableCredits.model_validate( WalletGetWithAvailableCredits.model_config["json_schema_extra"]["examples"][ @@ -76,14 +76,14 @@ def mock_licensed_items_purchase_functions(mocker: MockerFixture) -> tuple: ), ) mock_get_pricing_unit = mocker.patch( - "simcore_service_webserver.licenses._licensed_items_api.get_pricing_plan_unit", + "simcore_service_webserver.licenses._licensed_items_service.get_pricing_plan_unit", spec=True, return_value=PricingUnitGet.model_validate( PricingUnitGet.model_config["json_schema_extra"]["examples"][0] ), ) mock_create_licensed_item_purchase = mocker.patch( - "simcore_service_webserver.licenses._licensed_items_api.licensed_items_purchases.create_licensed_item_purchase", + "simcore_service_webserver.licenses._licensed_items_service.licensed_items_purchases.create_licensed_item_purchase", spec=True, ) @@ -106,7 +106,7 @@ async def test_licensed_items_purchase( ): assert client.app - licensed_item_db = await _licensed_items_db.create( + licensed_item_db = await _licensed_items_repository.create( client.app, product_name=osparc_product_name, name="Model A", diff --git a/services/web/server/tests/unit/with_dbs/04/licenses/test_licenses_rpc.py b/services/web/server/tests/unit/with_dbs/04/licenses/test_licenses_rpc.py index 7bbfae01c90..836a7fe05e6 100644 --- a/services/web/server/tests/unit/with_dbs/04/licenses/test_licenses_rpc.py +++ b/services/web/server/tests/unit/with_dbs/04/licenses/test_licenses_rpc.py @@ -26,7 +26,7 @@ from settings_library.rabbit import RabbitSettings from simcore_postgres_database.models.users import UserRole from simcore_service_webserver.application_settings import ApplicationSettings -from simcore_service_webserver.licenses import _licensed_items_db +from simcore_service_webserver.licenses import _licensed_items_repository pytest_simcore_core_services_selection = [ "rabbit", @@ -73,7 +73,7 @@ async def rpc_client( @pytest.fixture def mock_get_wallet_by_user(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.get_wallet_by_user", + "simcore_service_webserver.licenses._licensed_items_checkouts_service.get_wallet_by_user", spec=True, ) @@ -86,7 +86,7 @@ def mock_get_wallet_by_user(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_checkout_licensed_item(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.checkout_licensed_item", + "simcore_service_webserver.licenses._licensed_items_checkouts_service.licensed_items_checkouts.checkout_licensed_item", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_GET, ) @@ -95,7 +95,7 @@ def mock_checkout_licensed_item(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_get_licensed_item_checkout(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.get_licensed_item_checkout", + "simcore_service_webserver.licenses._licensed_items_checkouts_service.licensed_items_checkouts.get_licensed_item_checkout", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_GET, ) @@ -104,7 +104,7 @@ def mock_get_licensed_item_checkout(mocker: MockerFixture) -> tuple: @pytest.fixture def mock_release_licensed_item(mocker: MockerFixture) -> tuple: return mocker.patch( - "simcore_service_webserver.licenses._licensed_items_checkouts_api.licensed_items_checkouts.release_licensed_item", + "simcore_service_webserver.licenses._licensed_items_checkouts_service.licensed_items_checkouts.release_licensed_item", spec=True, return_value=_LICENSED_ITEM_CHECKOUT_GET, ) @@ -132,7 +132,7 @@ async def test_license_checkout_workflow( assert len(result.items) == 0 assert result.total == 0 - license_item_db = await _licensed_items_db.create( + license_item_db = await _licensed_items_repository.create( client.app, product_name=osparc_product_name, name="Model A",