Source code for galaxy.webapps.galaxy.services.sharable

import logging
from typing import (
    Dict,
    List,
    Optional,
    Set,
    Tuple,
    Type,
    Union,
)

from sqlalchemy import false

from galaxy.managers import base
from galaxy.managers.sharable import (
    SharableModelManager,
    SharableModelSerializer,
)
from galaxy.model import (
    History,
    Page,
    StoredWorkflow,
    User,
    Visualization,
)
from galaxy.schema.fields import DecodedDatabaseIdField
from galaxy.schema.notifications import (
    NewSharedItemNotificationContent,
    NotificationCreateData,
    NotificationCreateRequest,
    NotificationRecipients,
    PersonalNotificationCategory,
    SharableItemType,
)
from galaxy.schema.schema import (
    SetSlugPayload,
    ShareWithPayload,
    ShareWithStatus,
    SharingOptions,
    SharingStatus,
    UserIdentifier,
)
from galaxy.webapps.galaxy.services.notifications import NotificationService

log = logging.getLogger(__name__)

SharableItem = Union[
    History,
    StoredWorkflow,
    Visualization,
    Page,
]


[docs]class ShareableService: """ Provides the common logic used by the API to share *any* kind of resource with other users. The Manager class of the particular resource must implement the SharableModelManager and have a compatible SharableModelSerializer implementation. """ share_with_status_cls: Type[ShareWithStatus] = ShareWithStatus
[docs] def __init__( self, manager: SharableModelManager, serializer: SharableModelSerializer, notification_service: NotificationService, ) -> None: self.manager = manager self.serializer = serializer self.notification_service = notification_service
[docs] def set_slug(self, trans, id: DecodedDatabaseIdField, payload: SetSlugPayload): item = self._get_item_by_id(trans, id) self.manager.set_slug(item, payload.new_slug, trans.user)
[docs] def sharing(self, trans, id: DecodedDatabaseIdField) -> SharingStatus: """Gets the current sharing status of the item with the given id.""" item = self._get_item_by_id(trans, id) return self._get_sharing_status(trans, item)
[docs] def publish(self, trans, id: DecodedDatabaseIdField) -> SharingStatus: """Makes this item publicly accessible. If this item contains other elements they will be publicly accessible too. """ item = self._get_item_by_id(trans, id) self.manager.make_members_public(trans, item) self.manager.publish(item) return self._get_sharing_status(trans, item)
[docs] def unpublish(self, trans, id: DecodedDatabaseIdField) -> SharingStatus: item = self._get_item_by_id(trans, id) self.manager.unpublish(item) return self._get_sharing_status(trans, item)
[docs] def share_with_users(self, trans, id: DecodedDatabaseIdField, payload: ShareWithPayload) -> ShareWithStatus: item = self._get_item_by_id(trans, id) users, errors = self._get_users(trans, payload.user_ids) extra, users_to_notify = self._share_with_options(trans, item, users, errors, payload.share_option) base_status = self._get_sharing_status(trans, item) status = self.share_with_status_cls.model_construct(**base_status.model_dump(), extra=extra) status.errors.extend(errors) galaxy_url = str(trans.url_builder("/", qualified=True)).rstrip("/") if trans.url_builder else None self._send_notification_to_users(users_to_notify, item, status, galaxy_url) return status
def _share_with_options( self, trans, item, users: Set[User], errors: Set[str], share_option: Optional[SharingOptions] = None, ): new_users = None extra = self.manager.get_sharing_extra_information(trans, item, users, errors, share_option) if not extra or extra.can_share: _, new_users, _ = self.manager.update_current_sharing_with_users(item, users) extra = None return extra, new_users def _get_item_by_id(self, trans, id: DecodedDatabaseIdField): class_name = self.manager.model_class.__name__ item = base.get_object(trans, id, class_name, check_ownership=True, check_accessible=True, deleted=False) return item def _get_sharing_status(self, trans, item): status = self.serializer.serialize_to_view( item, user=trans.user, trans=trans, default_view="sharing", encode_id=False ) status["users_shared_with"] = [{"id": a.user.id, "email": a.user.email} for a in item.users_shared_with] return SharingStatus(**status) def _get_users(self, trans, emails_or_ids: List[UserIdentifier]) -> Tuple[Set[User], Set[str]]: send_to_users: Set[User] = set() send_to_err: Set[str] = set() for email_or_id in set(emails_or_ids): send_to_user = None if isinstance(email_or_id, int): send_to_user = self.manager.user_manager.by_id(email_or_id) if send_to_user.deleted: send_to_user = None else: email_address = email_or_id.strip() if not email_address: continue send_to_user = self.manager.user_manager.by_email( email_address, filters=[User.table.c.deleted == false()] ) if not send_to_user: send_to_err.add(f"{email_or_id} is not a valid Galaxy user.") elif send_to_user == trans.user: send_to_err.add("You cannot share resources with yourself.") else: send_to_users.add(send_to_user) return send_to_users, send_to_err def _send_notification_to_users( self, users_to_notify: Set[User], item: SharableItem, status: ShareWithStatus, galaxy_url: Optional[str] = None ): if ( self.notification_service.notification_manager.notifications_enabled and not status.errors and users_to_notify ): request = SharedItemNotificationFactory.build_notification_request( item, users_to_notify, status, galaxy_url ) # We can set force_sync=True here because we already have the set of users to notify # and there is no need to resolve them asynchronously as no groups or roles are involved. self.notification_service.send_notification_internal(request, force_sync=True)
[docs]class SharedItemNotificationFactory: source = "galaxy_sharing_system" type_map: Dict[Type[SharableItem], SharableItemType] = { History: "history", StoredWorkflow: "workflow", Visualization: "visualization", Page: "page", }
[docs] @staticmethod def build_notification_request( item: SharableItem, users_to_notify: Set[User], status: ShareWithStatus, galaxy_url: Optional[str] = None ) -> NotificationCreateRequest: user_ids = [user.id for user in users_to_notify] request = NotificationCreateRequest( recipients=NotificationRecipients.model_construct(user_ids=user_ids), notification=NotificationCreateData( source=SharedItemNotificationFactory.source, variant="info", category=PersonalNotificationCategory.new_shared_item, content=NewSharedItemNotificationContent( item_type=SharedItemNotificationFactory.type_map[type(item)], item_name=status.title, owner_name=status.username, slug=status.username_and_slug, ), ), galaxy_url=galaxy_url, ) return request