diff --git a/.github/sql-fixtures/fixtures.sql b/.github/sql-fixtures/fixtures.sql index 4450468c4..c923adf29 100644 Binary files a/.github/sql-fixtures/fixtures.sql and b/.github/sql-fixtures/fixtures.sql differ diff --git a/python/apps/taiga/src/taiga/attachments/repositories.py b/python/apps/taiga/src/taiga/attachments/repositories.py index e21c09c00..ea8d847af 100644 --- a/python/apps/taiga/src/taiga/attachments/repositories.py +++ b/python/apps/taiga/src/taiga/attachments/repositories.py @@ -69,7 +69,7 @@ async def _apply_prefetch_related_to_queryset( ########################################################## -# create comment +# create attachment ########################################################## diff --git a/python/apps/taiga/src/taiga/base/db/commands.py b/python/apps/taiga/src/taiga/base/db/commands.py index a88eaab9a..cb847517b 100644 --- a/python/apps/taiga/src/taiga/base/db/commands.py +++ b/python/apps/taiga/src/taiga/base/db/commands.py @@ -15,7 +15,6 @@ from django.conf import settings from taiga.base.django.commands import call_django_command from taiga.base.utils import pprint -from taiga.base.utils.commands import unwrap_typer_param cli = typer.Typer( name="The Taiga DB Manager", @@ -75,7 +74,6 @@ def init_migrations( @cli.command(help="Creates new migration(s) for apps.") -@unwrap_typer_param def make_migrations( name: str = typer.Option("", "--name", "-n", help="Use this name for migration file(s)."), dry_run: bool = typer.Option( diff --git a/python/apps/taiga/src/taiga/base/django/settings.py b/python/apps/taiga/src/taiga/base/django/settings.py index 3fb45be47..fd2fd525f 100644 --- a/python/apps/taiga/src/taiga/base/django/settings.py +++ b/python/apps/taiga/src/taiga/base/django/settings.py @@ -41,6 +41,7 @@ "taiga.commons.storage", "taiga.emails", "taiga.mediafiles", + "taiga.notifications", "taiga.projects.invitations", "taiga.projects.memberships", "taiga.projects.projects", diff --git a/python/apps/taiga/src/taiga/base/serializers/__init__.py b/python/apps/taiga/src/taiga/base/serializers/__init__.py index d602df583..0667b635a 100644 --- a/python/apps/taiga/src/taiga/base/serializers/__init__.py +++ b/python/apps/taiga/src/taiga/base/serializers/__init__.py @@ -7,7 +7,7 @@ from humps.main import camelize from pydantic import BaseModel as _BaseModel -from taiga.base.serializers.fields import UUIDB64 # noqa +from taiga.base.serializers.fields import UUIDB64, CamelizeDict, FileField # noqa class BaseModel(_BaseModel): diff --git a/python/apps/taiga/src/taiga/base/serializers/fields.py b/python/apps/taiga/src/taiga/base/serializers/fields.py index 596ca9597..d61b04a6b 100644 --- a/python/apps/taiga/src/taiga/base/serializers/fields.py +++ b/python/apps/taiga/src/taiga/base/serializers/fields.py @@ -5,9 +5,10 @@ # # Copyright (c) 2023-present Kaleidos INC -from typing import Any, Callable, Generator +from typing import Any, Callable, Generator, TypeVar from uuid import UUID +from humps.main import camelize from pydantic import AnyHttpUrl, AnyUrl, BaseConfig from pydantic.fields import ModelField from taiga.base.utils.uuid import encode_uuid_to_b64str @@ -17,8 +18,10 @@ class UUIDB64(UUID): @classmethod - def __modify_schema__(cls, field_schema: dict[str, Any]) -> None: - field_schema["example"] = "6JgsbGyoEe2VExhWgGrI2w" + def __modify_schema__(cls, field_schema: dict[str, Any] = {}) -> None: + field_schema.update( + example="6JgsbGyoEe2VExhWgGrI2w", + ) @classmethod def __get_validators__(cls) -> CallableGenerator: @@ -33,3 +36,26 @@ class FileField(AnyHttpUrl): @classmethod def validate(cls, value: Any, field: ModelField, config: BaseConfig) -> AnyUrl: return value.url + + +_Key = TypeVar("_Key") +_Val = TypeVar("_Val") + + +class CamelizeDict(str): + def __modify_schema__(cls, field_schema: dict[str, Any] = {}) -> None: + field_schema.update( + type="object", + example={}, + ) + + @classmethod + def __get_validators__(cls) -> CallableGenerator: + yield cls.validate + + @classmethod + def validate(cls, value: dict[_Key, _Val]) -> dict[_Key, _Val]: + return camelize(value) + + def __repr__(self) -> str: + return "CamelizeDict" diff --git a/python/apps/taiga/src/taiga/base/utils/commands.py b/python/apps/taiga/src/taiga/base/utils/commands.py index 2ccbd7e0e..d88331e44 100644 --- a/python/apps/taiga/src/taiga/base/utils/commands.py +++ b/python/apps/taiga/src/taiga/base/utils/commands.py @@ -5,13 +5,10 @@ # # Copyright (c) 2023-present Kaleidos INC -import functools import os from contextlib import contextmanager from pathlib import Path -from typing import Any, Callable, Generator - -import typer +from typing import Generator @contextmanager @@ -31,30 +28,3 @@ def set_working_directory(path: Path) -> Generator[None, None, None]: yield finally: os.chdir(origin) - - -def unwrap_typer_param(f: Callable[..., Any]) -> Callable[..., Any]: - """ - Unwraps the default values from typer.Argument or typer.Option to allow function to be called normally. - See: https://github.com/tiangolo/typer/issues/279 and https://gitlab.com/boratko/typer-utils - """ - if f.__defaults__ is None: - return f - else: - patched_defaults = [] - actual_default_observed = False - for i, value in enumerate(f.__defaults__): - default_value = value.default if isinstance(value, typer.models.ParameterInfo) else value - if default_value != ...: - actual_default_observed = True - patched_defaults.append(default_value) - elif actual_default_observed: - raise SyntaxError("non-default argument follows default argument") - f.__defaults__ = tuple(patched_defaults) - - @functools.wraps(f) - def wrapper(*args: Any, **kwargs: Any) -> Callable[..., Any]: - f.__defaults__ = tuple(patched_defaults) - return f(*args, **kwargs) - - return wrapper diff --git a/python/apps/taiga/src/taiga/base/validators/fields/i18n.py b/python/apps/taiga/src/taiga/base/validators/fields/i18n.py index 53aaa3c85..e58527b7e 100644 --- a/python/apps/taiga/src/taiga/base/validators/fields/i18n.py +++ b/python/apps/taiga/src/taiga/base/validators/fields/i18n.py @@ -16,8 +16,10 @@ class LanguageCode(str): @classmethod def __modify_schema__(cls: Type["LanguageCode"], field_schema: dict[str, Any]) -> None: - field_schema["example"] = settings.LANG - field_schema["enum"] = i18n.available_languages + field_schema.update( + example=settings.LANG, + enum=i18n.available_languages, + ) @classmethod def __get_validators__(cls: Type["LanguageCode"]) -> CallableGenerator: diff --git a/python/apps/taiga/src/taiga/base/validators/fields/uuid.py b/python/apps/taiga/src/taiga/base/validators/fields/uuid.py index e8bee6309..5929ee813 100644 --- a/python/apps/taiga/src/taiga/base/validators/fields/uuid.py +++ b/python/apps/taiga/src/taiga/base/validators/fields/uuid.py @@ -16,8 +16,10 @@ class B64UUID(UUID): @classmethod def __modify_schema__(cls: Type["B64UUID"], field_schema: dict[str, Any]) -> None: - field_schema["example"] = "6JgsbGyoEe2VExhWgGrI2w" - field_schema["format"] = None + field_schema.update( + example="6JgsbGyoEe2VExhWgGrI2w", + format=None, + ) @classmethod def __get_validators__(cls: Type["B64UUID"]) -> CallableGenerator: diff --git a/python/apps/taiga/src/taiga/notifications/__init__.py b/python/apps/taiga/src/taiga/notifications/__init__.py new file mode 100644 index 000000000..87d9a5256 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/__init__.py @@ -0,0 +1,6 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC diff --git a/python/apps/taiga/src/taiga/notifications/admin.py b/python/apps/taiga/src/taiga/notifications/admin.py new file mode 100644 index 000000000..89fab12c8 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/admin.py @@ -0,0 +1,23 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from typing import Any + +from taiga.base.db import admin +from taiga.base.db.admin.http import HttpRequest +from taiga.notifications.models import Notification + + +@admin.register(Notification) +class NotificationAdmin(admin.ModelAdmin[Notification]): + list_display = ("id", "type", "owner", "created_at", "read_at") + + def has_change_permission(self, request: HttpRequest, obj: Any = None) -> bool: + return False + + def has_add_permission(self, request: HttpRequest, obj: Any = None) -> bool: + return False diff --git a/python/apps/taiga/src/taiga/notifications/api.py b/python/apps/taiga/src/taiga/notifications/api.py new file mode 100644 index 000000000..991bcfc3b --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/api.py @@ -0,0 +1,57 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from taiga.base.api import AuthRequest +from taiga.base.api.permissions import check_permissions +from taiga.exceptions.api.errors import ERROR_403 +from taiga.notifications import services as notifications_services +from taiga.notifications.models import Notification +from taiga.notifications.serializers import NotificationCountersSerializer, NotificationSerializer +from taiga.permissions import IsAuthenticated +from taiga.routers import routes + +LIST_MY_NOTIFICATIONS = IsAuthenticated() +COUNT_MY_NOTIFICATIONS = IsAuthenticated() + +########################################################## +# list notifications +########################################################## + + +@routes.notifications.get( + "/my/notifications", + name="my.notifications.list", + summary="List all the user notifications", + responses=ERROR_403, + response_model=list[NotificationSerializer], +) +async def list_my_notifications(request: AuthRequest, read: bool | None = None) -> list[Notification]: + """ + List the notifications of the logged user. + """ + await check_permissions(permissions=LIST_MY_NOTIFICATIONS, user=request.user, obj=None) + return await notifications_services.list_user_notifications(user=request.user, is_read=read) + + +########################################################## +# count notifications +########################################################## + + +@routes.notifications.get( + "/my/notifications/count", + name="my.notifications.count", + summary="Get user notifications counters", + responses=ERROR_403, + response_model=NotificationCountersSerializer, +) +async def count_my_notifications(request: AuthRequest) -> dict[str, int]: + """ + Get user notifications counters + """ + await check_permissions(permissions=COUNT_MY_NOTIFICATIONS, user=request.user, obj=None) + return await notifications_services.count_user_notifications(user=request.user) diff --git a/python/apps/taiga/src/taiga/notifications/events/__init__.py b/python/apps/taiga/src/taiga/notifications/events/__init__.py new file mode 100644 index 000000000..8e08a1605 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/events/__init__.py @@ -0,0 +1,25 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from taiga.events import events_manager +from taiga.notifications.events.content import CreateNotificationContent +from taiga.notifications.models import Notification + +CREATE_NOTIFICATION = "notifications.create" + + +async def emit_event_when_notifications_are_created( + notifications: list[Notification], +) -> None: + for notification in notifications: + await events_manager.publish_on_user_channel( + user=notification.owner, + type=CREATE_NOTIFICATION, + content=CreateNotificationContent( + notification=notification, + ), + ) diff --git a/python/apps/taiga/src/taiga/notifications/events/content.py b/python/apps/taiga/src/taiga/notifications/events/content.py new file mode 100644 index 000000000..f89ce4c82 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/events/content.py @@ -0,0 +1,13 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from taiga.base.serializers import BaseModel +from taiga.notifications.serializers import NotificationSerializer + + +class CreateNotificationContent(BaseModel): + notification: NotificationSerializer diff --git a/python/apps/taiga/src/taiga/notifications/migrations/0001_initial.py b/python/apps/taiga/src/taiga/notifications/migrations/0001_initial.py new file mode 100644 index 000000000..ba759eb98 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/migrations/0001_initial.py @@ -0,0 +1,89 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +# Generated by Django 4.2.3 on 2023-10-04 18:15 + +import django.contrib.postgres.fields.jsonb +import django.db.models.deletion +import taiga.base.db.models +import taiga.base.utils.datetime +from django.conf import settings +from django.db import migrations, models + + +class Migration(migrations.Migration): + initial = True + + dependencies = [ + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ] + + operations = [ + migrations.CreateModel( + name="Notification", + fields=[ + ( + "id", + models.UUIDField( + blank=True, + default=taiga.base.db.models.uuid_generator, + editable=False, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ( + "created_at", + models.DateTimeField( + default=taiga.base.utils.datetime.aware_utcnow, + verbose_name="created at", + ), + ), + ("type", models.CharField(max_length=500, verbose_name="type")), + ( + "read_at", + models.DateTimeField(blank=True, null=True, verbose_name="read at"), + ), + ( + "content", + django.contrib.postgres.fields.jsonb.JSONField(default=dict, verbose_name="content"), + ), + ( + "created_by", + models.ForeignKey( + blank=True, + null=True, + on_delete=django.db.models.deletion.SET_NULL, + to=settings.AUTH_USER_MODEL, + verbose_name="created by", + ), + ), + ( + "owner", + models.ForeignKey( + on_delete=django.db.models.deletion.CASCADE, + related_name="notifications", + to=settings.AUTH_USER_MODEL, + verbose_name="owner", + ), + ), + ], + options={ + "verbose_name": "notification", + "verbose_name_plural": "notifications", + "ordering": ["-created_at"], + "indexes": [ + models.Index(fields=["owner"], name="notificatio_owner_i_2bc47d_idx"), + models.Index( + fields=["owner", "read_at"], + name="notificatio_owner_i_37308f_idx", + ), + ], + }, + ), + ] diff --git a/python/apps/taiga/src/taiga/notifications/migrations/__init__.py b/python/apps/taiga/src/taiga/notifications/migrations/__init__.py new file mode 100644 index 000000000..87d9a5256 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/migrations/__init__.py @@ -0,0 +1,6 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC diff --git a/python/apps/taiga/src/taiga/notifications/models.py b/python/apps/taiga/src/taiga/notifications/models.py new file mode 100644 index 000000000..e0dbd1b6e --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/models.py @@ -0,0 +1,54 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from taiga.base.db import models +from taiga.base.db.mixins import CreatedMetaInfoMixin + +####################################################################### +# Base Notification +###################################################################### + + +class Notification(models.BaseModel, CreatedMetaInfoMixin): + type = models.CharField( + max_length=500, + null=False, + blank=False, + verbose_name="type", + ) + owner = models.ForeignKey( + "users.User", + null=False, + blank=False, + on_delete=models.CASCADE, + related_name="notifications", + verbose_name="owner", + ) + read_at = models.DateTimeField( + null=True, + blank=True, + verbose_name="read at", + ) + content = models.JSONField( + null=False, + blank=False, + default=dict, + verbose_name="content", + ) + + class Meta: + verbose_name = "notification" + verbose_name_plural = "notifications" + ordering = ["-created_at"] + indexes = [ + models.Index( + fields=[ + "owner", + ] + ), + models.Index(fields=["owner", "read_at"]), + ] diff --git a/python/apps/taiga/src/taiga/notifications/repositories.py b/python/apps/taiga/src/taiga/notifications/repositories.py new file mode 100644 index 000000000..a2d80ad44 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/repositories.py @@ -0,0 +1,92 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from collections.abc import Iterable +from typing import Any, TypedDict + +from taiga.base.db.models import QuerySet +from taiga.notifications.models import Notification +from taiga.users.models import User + +########################################################## +# filters and querysets +########################################################## + +DEFAULT_QUERYSET = Notification.objects.select_related("created_by").all() + + +class NotificationFilters(TypedDict, total=False): + owner: User + is_read: bool + + +async def _apply_filters_to_queryset( + qs: QuerySet[Notification], + filters: NotificationFilters = {}, +) -> QuerySet[Notification]: + filter_data = dict(filters.copy()) + + if "is_read" in filter_data: + is_read = filter_data.pop("is_read") + filter_data["read_at__isnull"] = not is_read + + return qs.filter(**filter_data) + + +########################################################## +# create notifications +########################################################## + + +async def create_notifications( + owners: Iterable[User], + created_by: User, + notification_type: str, + content: dict[str, Any], +) -> list[Notification]: + notifications = [ + Notification( + owner=owner, + created_by=created_by, + type=notification_type, + content=content, + ) + for owner in owners + ] + + return await Notification.objects.abulk_create(notifications) + + +########################################################## +# list notifications +########################################################## + + +async def list_notifications( + filters: NotificationFilters = {}, + offset: int | None = None, + limit: int | None = None, +) -> list[Notification]: + qs = await _apply_filters_to_queryset(qs=DEFAULT_QUERYSET, filters=filters) + + if limit is not None and offset is not None: + limit += offset + + return [a async for a in qs[offset:limit]] + + +########################################################## +# misc +########################################################## + + +async def count_notifications( + filters: NotificationFilters = {}, +) -> int: + qs = await _apply_filters_to_queryset(qs=DEFAULT_QUERYSET, filters=filters) + + return await qs.acount() diff --git a/python/apps/taiga/src/taiga/notifications/serializers.py b/python/apps/taiga/src/taiga/notifications/serializers.py new file mode 100644 index 000000000..2ed9eb332 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/serializers.py @@ -0,0 +1,29 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from datetime import datetime + +from taiga.base.serializers import UUIDB64, BaseModel, CamelizeDict +from taiga.users.serializers.nested import UserNestedSerializer + + +class NotificationSerializer(BaseModel): + id: UUIDB64 + type: str + created_by: UserNestedSerializer | None + created_at: datetime + read_at: datetime | None + content: CamelizeDict + + class Config: + orm_mode = True + + +class NotificationCountersSerializer(BaseModel): + read: int + unread: int + total: int diff --git a/python/apps/taiga/src/taiga/notifications/services.py b/python/apps/taiga/src/taiga/notifications/services.py new file mode 100644 index 000000000..a0e970935 --- /dev/null +++ b/python/apps/taiga/src/taiga/notifications/services.py @@ -0,0 +1,39 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC +from collections.abc import Iterable + +from taiga.base.serializers import BaseModel +from taiga.notifications import events as notifications_events +from taiga.notifications import repositories as notifications_repositories +from taiga.notifications.models import Notification +from taiga.notifications.repositories import NotificationFilters +from taiga.users.models import User + + +async def notify_users(type: str, emitted_by: User, notified_users: Iterable[User], content: BaseModel) -> None: + notifications = await notifications_repositories.create_notifications( + owners=notified_users, + created_by=emitted_by, + notification_type=type, + content=content.dict(), + ) + await notifications_events.emit_event_when_notifications_are_created(notifications=notifications) + + +async def list_user_notifications(user: User, is_read: bool | None = None) -> list[Notification]: + filters: NotificationFilters = {"owner": user} + + if is_read is not None: + filters["is_read"] = is_read + + return await notifications_repositories.list_notifications(filters=filters) + + +async def count_user_notifications(user: User) -> dict[str, int]: + total = await notifications_repositories.count_notifications(filters={"owner": user}) + read = await notifications_repositories.count_notifications(filters={"owner": user, "is_read": True}) + return {"total": total, "read": read, "unread": total - read} diff --git a/python/apps/taiga/src/taiga/projects/projects/serializers/nested.py b/python/apps/taiga/src/taiga/projects/projects/serializers/nested.py index 15c826f36..888a2b7c1 100644 --- a/python/apps/taiga/src/taiga/projects/projects/serializers/nested.py +++ b/python/apps/taiga/src/taiga/projects/projects/serializers/nested.py @@ -20,6 +20,15 @@ class Config: orm_mode = True +class ProjectLinkNestedSerializer(BaseModel): + id: UUIDB64 + name: str + slug: str + + class Config: + orm_mode = True + + class ProjectSmallNestedSerializer(BaseModel): id: UUIDB64 name: str diff --git a/python/apps/taiga/src/taiga/routers/loader.py b/python/apps/taiga/src/taiga/routers/loader.py index 188c8e786..4841f3575 100644 --- a/python/apps/taiga/src/taiga/routers/loader.py +++ b/python/apps/taiga/src/taiga/routers/loader.py @@ -13,6 +13,7 @@ from taiga.integrations.github.auth import api as github_auth_api # noqa from taiga.integrations.gitlab.auth import api as gitlab_auth_api # noqa from taiga.integrations.google.auth import api as google_auth_api # noqa +from taiga.notifications import api as notifications_api # noqa from taiga.projects.invitations import api as projects_invitations_api # noqa from taiga.projects.memberships import api as projects_memberships_api # noqa from taiga.projects.projects import api as projects_api # noqa @@ -39,6 +40,7 @@ def load_routes(api: FastAPI) -> None: api.include_router(routes.auth) api.include_router(routes.users) api.include_router(routes.unauth_users) + api.include_router(routes.notifications) api.include_router(routes.workspaces) api.include_router(routes.workspaces_invitations) api.include_router(routes.unauth_workspaces_invitations) diff --git a/python/apps/taiga/src/taiga/routers/routes.py b/python/apps/taiga/src/taiga/routers/routes.py index 3ad4cad44..4232bb2a9 100644 --- a/python/apps/taiga/src/taiga/routers/routes.py +++ b/python/apps/taiga/src/taiga/routers/routes.py @@ -32,6 +32,16 @@ } ) +# /notifications +notifications = AuthAPIRouter(tags=["notifications"]) +tags_metadata.append( + { + "name": "notifications", + "description": "Endpoint for user notifications resources.", + } +) + + # /workspaces workspaces = AuthAPIRouter(tags=["workspaces"]) tags_metadata.append( diff --git a/python/apps/taiga/src/taiga/stories/assignments/api/__init__.py b/python/apps/taiga/src/taiga/stories/assignments/api/__init__.py index 991da00e1..5d2ecd659 100644 --- a/python/apps/taiga/src/taiga/stories/assignments/api/__init__.py +++ b/python/apps/taiga/src/taiga/stories/assignments/api/__init__.py @@ -51,7 +51,10 @@ async def create_story_assignment( await check_permissions(permissions=CREATE_STORY_ASSIGNMENT, user=request.user, obj=story) return await story_assignments_services.create_story_assignment( - project_id=project_id, story=story, username=form.username + project_id=project_id, + story=story, + username=form.username, + created_by=request.user, ) @@ -77,9 +80,12 @@ async def delete_story_assignment( Delete a story assignment """ story_assignment = await get_story_assignment_or_404(project_id, ref, username) + story = await get_story_or_404(project_id, ref) await check_permissions(permissions=DELETE_STORY_ASSIGNMENT, user=request.user, obj=story_assignment.story) - await story_assignments_services.delete_story_assignment(story_assignment=story_assignment) + await story_assignments_services.delete_story_assignment( + story_assignment=story_assignment, story=story, deleted_by=request.user + ) ################################################ diff --git a/python/apps/taiga/src/taiga/stories/assignments/notifications/__init__.py b/python/apps/taiga/src/taiga/stories/assignments/notifications/__init__.py new file mode 100644 index 000000000..e302ce74e --- /dev/null +++ b/python/apps/taiga/src/taiga/stories/assignments/notifications/__init__.py @@ -0,0 +1,55 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from taiga.notifications import services as notifications_services +from taiga.stories.assignments.notifications.content import ( + StoryAssignNotificationContent, + StoryUnassignNotificationContent, +) +from taiga.stories.stories.models import Story +from taiga.users.models import User + +STORIES_ASSIGN = "stories.assign" +STORIES_UNASSIGN = "stories.unassign" + + +async def notify_when_story_is_assigned(story: Story, assigned_to: User, emitted_by: User) -> None: + """ + Emit notification when a story is assigned. + """ + notified_users = {assigned_to} + if story.created_by: + notified_users.add(story.created_by) + notified_users.discard(emitted_by) + + await notifications_services.notify_users( + type=STORIES_ASSIGN, + emitted_by=emitted_by, + notified_users=notified_users, + content=StoryAssignNotificationContent( + project=story.project, story=story, assigned_by=emitted_by, assigned_to=assigned_to + ), + ) + + +async def notify_when_story_is_unassigned(story: Story, unassigned_to: User, emitted_by: User) -> None: + """ + Emit notification when story is unassigned. + """ + notified_users = {unassigned_to} + if story.created_by: + notified_users.add(story.created_by) + notified_users.discard(emitted_by) + + await notifications_services.notify_users( + type=STORIES_UNASSIGN, + emitted_by=emitted_by, + notified_users=notified_users, + content=StoryUnassignNotificationContent( + project=story.project, story=story, unassigned_by=emitted_by, unassigned_to=unassigned_to + ), + ) diff --git a/python/apps/taiga/src/taiga/stories/assignments/notifications/content.py b/python/apps/taiga/src/taiga/stories/assignments/notifications/content.py new file mode 100644 index 000000000..1891e5a1c --- /dev/null +++ b/python/apps/taiga/src/taiga/stories/assignments/notifications/content.py @@ -0,0 +1,31 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from taiga.base.serializers import BaseModel +from taiga.projects.projects.serializers.nested import ProjectLinkNestedSerializer +from taiga.stories.stories.serializers.nested import StoryNestedSerializer +from taiga.users.serializers.nested import UserNestedSerializer + + +class StoryAssignNotificationContent(BaseModel): + project: ProjectLinkNestedSerializer + story: StoryNestedSerializer + assigned_by: UserNestedSerializer + assigned_to: UserNestedSerializer + + class Config: + orm_mode = True + + +class StoryUnassignNotificationContent(BaseModel): + project: ProjectLinkNestedSerializer + story: StoryNestedSerializer + unassigned_by: UserNestedSerializer + unassigned_to: UserNestedSerializer + + class Config: + orm_mode = True diff --git a/python/apps/taiga/src/taiga/stories/assignments/services/__init__.py b/python/apps/taiga/src/taiga/stories/assignments/services/__init__.py index 091d9cb69..33adba0f5 100644 --- a/python/apps/taiga/src/taiga/stories/assignments/services/__init__.py +++ b/python/apps/taiga/src/taiga/stories/assignments/services/__init__.py @@ -9,17 +9,19 @@ from taiga.projects.memberships import repositories as pj_memberships_repositories from taiga.stories.assignments import events as stories_assignments_events +from taiga.stories.assignments import notifications as stories_assignments_notifications from taiga.stories.assignments import repositories as story_assignments_repositories from taiga.stories.assignments.models import StoryAssignment from taiga.stories.assignments.services import exceptions as ex from taiga.stories.stories.models import Story +from taiga.users.models import User ########################################################## # create story assignment ########################################################## -async def create_story_assignment(project_id: UUID, story: Story, username: str) -> StoryAssignment: +async def create_story_assignment(project_id: UUID, story: Story, username: str, created_by: User) -> StoryAssignment: pj_membership = await pj_memberships_repositories.get_project_membership( filters={"project_id": project_id, "username": username, "permissions": ["view_story"]} ) @@ -31,6 +33,9 @@ async def create_story_assignment(project_id: UUID, story: Story, username: str) story_assignment, created = await story_assignments_repositories.create_story_assignment(story=story, user=user) if created: await stories_assignments_events.emit_event_when_story_assignment_is_created(story_assignment=story_assignment) + await stories_assignments_notifications.notify_when_story_is_assigned( + story=story, assigned_to=user, emitted_by=created_by + ) return story_assignment @@ -52,9 +57,12 @@ async def get_story_assignment(project_id: UUID, ref: int, username: str) -> Sto ########################################################## -async def delete_story_assignment(story_assignment: StoryAssignment) -> bool: +async def delete_story_assignment(story_assignment: StoryAssignment, story: Story, deleted_by: User) -> bool: deleted = await story_assignments_repositories.delete_stories_assignments(filters={"id": story_assignment.id}) if deleted > 0: await stories_assignments_events.emit_event_when_story_assignment_is_deleted(story_assignment=story_assignment) + await stories_assignments_notifications.notify_when_story_is_unassigned( + story=story, unassigned_to=story_assignment.user, emitted_by=deleted_by + ) return True return False diff --git a/python/apps/taiga/src/taiga/stories/stories/services/__init__.py b/python/apps/taiga/src/taiga/stories/stories/services/__init__.py index eb771fb81..4aabcd5ac 100644 --- a/python/apps/taiga/src/taiga/stories/stories/services/__init__.py +++ b/python/apps/taiga/src/taiga/stories/stories/services/__init__.py @@ -109,7 +109,7 @@ async def list_paginated_stories( async def get_story(project_id: UUID, ref: int) -> Story | None: return await stories_repositories.get_story( filters={"ref": ref, "project_id": project_id}, - select_related=["project", "workspace", "workflow"], + select_related=["project", "workspace", "workflow", "created_by"], ) @@ -362,4 +362,4 @@ async def delete_story(story: Story, deleted_by: AnyUser) -> bool: ) return True - return False + return False \ No newline at end of file diff --git a/python/apps/taiga/tests/integration/taiga/notifications/__init__.py b/python/apps/taiga/tests/integration/taiga/notifications/__init__.py new file mode 100644 index 000000000..87d9a5256 --- /dev/null +++ b/python/apps/taiga/tests/integration/taiga/notifications/__init__.py @@ -0,0 +1,6 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC diff --git a/python/apps/taiga/tests/integration/taiga/notifications/test_api.py b/python/apps/taiga/tests/integration/taiga/notifications/test_api.py new file mode 100644 index 000000000..b65875271 --- /dev/null +++ b/python/apps/taiga/tests/integration/taiga/notifications/test_api.py @@ -0,0 +1,86 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +import pytest +from fastapi import status +from taiga.base.utils.datetime import aware_utcnow +from tests.utils import factories as f + +pytestmark = pytest.mark.django_db(transaction=True) + +########################################################## +# GET my/notifications +########################################################## + + +async def test_list_my_notifications_200_ok(client): + user = await f.create_user() + await f.create_notification(owner=user) + await f.create_notification(owner=user) + await f.create_notification(owner=user, read_at=aware_utcnow()) + + client.login(user) + response = client.get("/my/notifications") + assert response.status_code == status.HTTP_200_OK, response.text + assert len(response.json()) == 3 + + +async def test_list_my_notifications_200_ok_filter_only_read(client): + user = await f.create_user() + await f.create_notification(owner=user) + await f.create_notification(owner=user) + await f.create_notification(owner=user, read_at=aware_utcnow()) + + client.login(user) + response = client.get("/my/notifications", params={"read": True}) + assert response.status_code == status.HTTP_200_OK, response.text + assert len(response.json()) == 1 + + +async def test_list_my_notifications_200_ok_filter_only_unread(client): + user = await f.create_user() + await f.create_notification(owner=user) + await f.create_notification(owner=user) + await f.create_notification(owner=user, read_at=aware_utcnow()) + + client.login(user) + response = client.get("/my/notifications", params={"read": False}) + assert response.status_code == status.HTTP_200_OK, response.text + assert len(response.json()) == 2 + + +async def test_list_my_notifications_403_forbidden_error(client): + user = await f.create_user() + await f.create_notification(owner=user) + + response = client.get("/my/notifications") + assert response.status_code == status.HTTP_403_FORBIDDEN, response.text + + +########################################################## +# GET my/notifications/count +########################################################## + + +async def test_count_my_notifications_200_ok(client): + user = await f.create_user() + await f.create_notification(owner=user) + await f.create_notification(owner=user) + await f.create_notification(owner=user, read_at=aware_utcnow()) + + client.login(user) + response = client.get("/my/notifications/count") + assert response.status_code == status.HTTP_200_OK, response.text + assert response.json() == {"total": 3, "read": 1, "unread": 2} + + +async def test_count_my_notifications_403_forbidden_error(client): + user = await f.create_user() + await f.create_notification(owner=user) + + response = client.get("/my/notifications/count") + assert response.status_code == status.HTTP_403_FORBIDDEN, response.text diff --git a/python/apps/taiga/tests/integration/taiga/notifications/test_repositories.py b/python/apps/taiga/tests/integration/taiga/notifications/test_repositories.py new file mode 100644 index 000000000..34274a7a6 --- /dev/null +++ b/python/apps/taiga/tests/integration/taiga/notifications/test_repositories.py @@ -0,0 +1,86 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +import pytest +from taiga.base.utils.datetime import aware_utcnow +from taiga.notifications import repositories +from tests.utils import factories as f + +pytestmark = pytest.mark.django_db(transaction=True) + + +########################################################## +# create notifications +########################################################## + + +async def test_create_notification(): + user1 = await f.create_user() + user2 = await f.create_user() + user3 = await f.create_user() + + notifications = await repositories.create_notifications( + owners=[user1, user2], + created_by=user3, + notification_type="test_notification", + content={"msg": "test"}, + ) + + assert len(notifications) == 2 + assert notifications[0].created_by == notifications[1].created_by == user3 + assert notifications[0].type == notifications[1].type == "test_notification" + assert notifications[0].content == notifications[1].content == {"msg": "test"} + assert notifications[0].owner == user1 + assert notifications[1].owner == user2 + + +########################################################## +# list notifications +########################################################## + + +async def test_list_notifications_filters(): + user1 = await f.create_user() + user2 = await f.create_user() + user3 = await f.create_user() + + n11 = await f.create_notification(owner=user1, created_by=user3) + n12 = await f.create_notification(owner=user1, created_by=user3, read_at=aware_utcnow()) + n13 = await f.create_notification(owner=user1, created_by=user3) + + n21 = await f.create_notification(owner=user2, created_by=user3) + n22 = await f.create_notification(owner=user2, created_by=user3, read_at=aware_utcnow()) + + assert [n22, n21, n13, n12, n11] == await repositories.list_notifications() + assert [n13, n12, n11] == await repositories.list_notifications(filters={"owner": user1}) + assert [n13, n11] == await repositories.list_notifications(filters={"owner": user1, "is_read": False}) + assert [n12] == await repositories.list_notifications(filters={"owner": user1, "is_read": True}) + assert [n22, n12] == await repositories.list_notifications(filters={"is_read": True}) + + +########################################################## +# misc +########################################################## + + +async def test_count_notifications(): + user1 = await f.create_user() + user2 = await f.create_user() + user3 = await f.create_user() + + await f.create_notification(owner=user1, created_by=user3) + await f.create_notification(owner=user1, created_by=user3, read_at=aware_utcnow()) + await f.create_notification(owner=user1, created_by=user3) + + await f.create_notification(owner=user2, created_by=user3) + await f.create_notification(owner=user2, created_by=user3, read_at=aware_utcnow()) + + assert 5 == await repositories.count_notifications() + assert 3 == await repositories.count_notifications(filters={"owner": user1}) + assert 2 == await repositories.count_notifications(filters={"owner": user1, "is_read": False}) + assert 1 == await repositories.count_notifications(filters={"owner": user1, "is_read": True}) + assert 2 == await repositories.count_notifications(filters={"is_read": True}) diff --git a/python/apps/taiga/tests/unit/taiga/notifications/__init__.py b/python/apps/taiga/tests/unit/taiga/notifications/__init__.py new file mode 100644 index 000000000..87d9a5256 --- /dev/null +++ b/python/apps/taiga/tests/unit/taiga/notifications/__init__.py @@ -0,0 +1,6 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC diff --git a/python/apps/taiga/tests/unit/taiga/notifications/test_services.py b/python/apps/taiga/tests/unit/taiga/notifications/test_services.py new file mode 100644 index 000000000..4fe1eb13e --- /dev/null +++ b/python/apps/taiga/tests/unit/taiga/notifications/test_services.py @@ -0,0 +1,109 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from unittest.mock import call, patch + +from taiga.base.serializers import BaseModel +from taiga.notifications import services +from tests.utils import factories as f + + +class SampleContent(BaseModel): + msg: str + + +##################################################################### +# notify_users +##################################################################### + + +async def test_notify_users(): + user = f.build_user() + notification = f.build_notification(type="test", owner=user) + content = SampleContent(msg="Test notify") + + with ( + patch( + "taiga.notifications.services.notifications_repositories", autospec=True + ) as fake_notifications_repository, + patch("taiga.notifications.services.notifications_events", autospec=True) as fake_notifications_events, + ): + fake_notifications_repository.create_notifications.return_value = [notification] + + await services.notify_users(type="test", emitted_by=user, notified_users=[user], content=content) + + fake_notifications_repository.create_notifications.assert_called_once_with( + owners=[user], created_by=user, notification_type="test", content={"msg": "Test notify"} + ) + + fake_notifications_events.emit_event_when_notifications_are_created.assert_called_once_with( + notifications=[notification] + ) + + +##################################################################### +# list_user_notifications +##################################################################### + + +async def test_list_user_notifications(): + user = f.build_user() + + with patch( + "taiga.notifications.services.notifications_repositories", autospec=True + ) as fake_notifications_repository: + await services.list_user_notifications(user=user) + + fake_notifications_repository.list_notifications.assert_called_once_with(filters={"owner": user}) + + +async def test_list_user_notifications_read_only(): + user = f.build_user() + + with patch( + "taiga.notifications.services.notifications_repositories", autospec=True + ) as fake_notifications_repository: + await services.list_user_notifications(user=user, is_read=True) + + fake_notifications_repository.list_notifications.assert_called_once_with( + filters={"owner": user, "is_read": True} + ) + + +async def test_list_user_notifications_unread_only(): + user = f.build_user() + + with patch( + "taiga.notifications.services.notifications_repositories", autospec=True + ) as fake_notifications_repository: + await services.list_user_notifications(user=user, is_read=False) + + fake_notifications_repository.list_notifications.assert_called_once_with( + filters={"owner": user, "is_read": False} + ) + + +##################################################################### +# count_user_notifications +##################################################################### + + +async def test_count_user_notifications(): + user = f.build_user() + + with patch( + "taiga.notifications.services.notifications_repositories", autospec=True + ) as fake_notifications_repository: + fake_notifications_repository.count_notifications.side_effect = [10, 2] + + result = await services.count_user_notifications(user=user) + + assert result == {"total": 10, "read": 2, "unread": 8} + + fake_notifications_repository.count_notifications.assert_has_awaits( + [call(filters={"owner": user}), call(filters={"owner": user, "is_read": True})] + ) diff --git a/python/apps/taiga/tests/unit/taiga/stories/assignments/test_services.py b/python/apps/taiga/tests/unit/taiga/stories/assignments/test_services.py index d8bb9d8d5..cf60e226f 100644 --- a/python/apps/taiga/tests/unit/taiga/stories/assignments/test_services.py +++ b/python/apps/taiga/tests/unit/taiga/stories/assignments/test_services.py @@ -33,7 +33,9 @@ async def test_create_story_assignment_not_member(): ) as fake_story_assignment_repo, pytest.raises(ex.InvalidAssignmentError), ): - await services.create_story_assignment(project_id=story.project.id, story=story, username=user.username) + await services.create_story_assignment( + project_id=story.project.id, story=story, username=user.username, created_by=story.created_by + ) fake_story_assignment_repo.create_story_assignment.assert_not_awaited() @@ -55,14 +57,20 @@ async def test_create_story_assignment_user_without_view_story_permission(): patch( "taiga.stories.assignments.services.stories_assignments_events", autospec=True ) as fake_stories_assignments_events, + patch( + "taiga.stories.assignments.services.stories_assignments_notifications", autospec=True + ) as fake_stories_assignments_notifications, pytest.raises(ex.InvalidAssignmentError), ): fake_pj_memberships_repo.get_project_membership.return_value = None fake_story_assignment_repo.create_story_assignment.return_value = None, False - await services.create_story_assignment(project_id=story.project.id, story=story, username=user.username) - fake_stories_assignments_events.emit_event_when_story_assignment_is_created.assert_not_awaited() + await services.create_story_assignment( + project_id=story.project.id, story=story, username=user.username, created_by=story.created_by + ) fake_story_assignment_repo.create_story_assignment.assert_not_awaited() + fake_stories_assignments_events.emit_event_when_story_assignment_is_created.assert_not_awaited() + fake_stories_assignments_notifications.notify_when_story_is_assigned.assert_not_awaited() async def test_create_story_assignment_ok(): @@ -83,18 +91,26 @@ async def test_create_story_assignment_ok(): patch( "taiga.stories.assignments.services.stories_assignments_events", autospec=True ) as fake_stories_assignments_events, + patch( + "taiga.stories.assignments.services.stories_assignments_notifications", autospec=True + ) as fake_stories_assignments_notifications, ): fake_pj_memberships_repo.get_project_membership.return_value = membership fake_story_assignment_repo.create_story_assignment.return_value = story_assignment, True - await services.create_story_assignment(project_id=project.id, story=story, username=user.username) - fake_stories_assignments_events.emit_event_when_story_assignment_is_created.assert_awaited_once_with( - story_assignment=story_assignment + await services.create_story_assignment( + project_id=project.id, story=story, username=user.username, created_by=story.created_by ) fake_story_assignment_repo.create_story_assignment.assert_awaited_once_with( story=story, user=user, ) + fake_stories_assignments_events.emit_event_when_story_assignment_is_created.assert_awaited_once_with( + story_assignment=story_assignment + ) + fake_stories_assignments_notifications.notify_when_story_is_assigned.assert_awaited_once_with( + story=story, assigned_to=user, emitted_by=story.created_by + ) async def test_create_story_assignment_already_assignment(): @@ -115,22 +131,32 @@ async def test_create_story_assignment_already_assignment(): patch( "taiga.stories.assignments.services.stories_assignments_events", autospec=True ) as fake_stories_assignments_events, + patch( + "taiga.stories.assignments.services.stories_assignments_notifications", autospec=True + ) as fake_stories_assignments_notifications, ): fake_pj_memberships_repo.get_project_membership.return_value = membership fake_story_assignment_repo.create_story_assignment.return_value = story_assignment, True - await services.create_story_assignment(project_id=project.id, story=story, username=user.username) + await services.create_story_assignment( + project_id=project.id, story=story, username=user.username, created_by=story.created_by + ) fake_story_assignment_repo.create_story_assignment.return_value = story_assignment, False - await services.create_story_assignment(project_id=project.id, story=story, username=user.username) - fake_stories_assignments_events.emit_event_when_story_assignment_is_created.assert_awaited_once_with( - story_assignment=story_assignment + await services.create_story_assignment( + project_id=project.id, story=story, username=user.username, created_by=story.created_by ) fake_story_assignment_repo.create_story_assignment.assert_awaited_with( story=story, user=user, ) + fake_stories_assignments_events.emit_event_when_story_assignment_is_created.assert_awaited_once_with( + story_assignment=story_assignment + ) + fake_stories_assignments_notifications.notify_when_story_is_assigned.assert_awaited_once_with( + story=story, assigned_to=user, emitted_by=story.created_by + ) ####################################################### @@ -175,15 +201,20 @@ async def test_delete_story_assignment_fail(): patch( "taiga.stories.assignments.services.stories_assignments_events", autospec=True ) as fake_stories_assignments_events, + patch( + "taiga.stories.assignments.services.stories_assignments_notifications", autospec=True + ) as fake_stories_assignments_notifications, ): fake_story_assignment_repo.delete_stories_assignments.return_value = 0 - await services.delete_story_assignment(story_assignment=story_assignment) - fake_stories_assignments_events.emit_event_when_story_assignment_is_deleted.assert_not_awaited() - + await services.delete_story_assignment( + story=story, story_assignment=story_assignment, deleted_by=story.created_by + ) fake_story_assignment_repo.delete_stories_assignments.assert_awaited_once_with( filters={"id": story_assignment.id}, ) + fake_stories_assignments_events.emit_event_when_story_assignment_is_deleted.assert_not_awaited() + fake_stories_assignments_notifications.notify_when_story_is_unassigned.assert_not_awaited() async def test_delete_story_assignment_ok(): @@ -198,14 +229,21 @@ async def test_delete_story_assignment_ok(): patch( "taiga.stories.assignments.services.stories_assignments_events", autospec=True ) as fake_stories_assignments_events, + patch( + "taiga.stories.assignments.services.stories_assignments_notifications", autospec=True + ) as fake_stories_assignments_notifications, ): fake_story_assignment_repo.delete_stories_assignments.return_value = 1 - await services.delete_story_assignment(story_assignment=story_assignment) - fake_stories_assignments_events.emit_event_when_story_assignment_is_deleted.assert_awaited_once_with( - story_assignment=story_assignment + await services.delete_story_assignment( + story=story, story_assignment=story_assignment, deleted_by=story.created_by ) - fake_story_assignment_repo.delete_stories_assignments.assert_awaited_once_with( filters={"id": story_assignment.id}, ) + fake_stories_assignments_events.emit_event_when_story_assignment_is_deleted.assert_awaited_once_with( + story_assignment=story_assignment + ) + fake_stories_assignments_notifications.notify_when_story_is_unassigned.assert_awaited_once_with( + story=story, unassigned_to=user, emitted_by=story.created_by + ) diff --git a/python/apps/taiga/tests/utils/factories/__init__.py b/python/apps/taiga/tests/utils/factories/__init__.py index 85296ce32..f69b15f47 100644 --- a/python/apps/taiga/tests/utils/factories/__init__.py +++ b/python/apps/taiga/tests/utils/factories/__init__.py @@ -19,6 +19,7 @@ build_string_uploadfile, ) from .mediafiles import MediafileFactory, build_mediafile, create_mediafile # noqa +from .notifications import NotificationFactory, build_notification, create_notification # noqa from .projects import ( # noqa ProjectFactory, ProjectInvitationFactory, diff --git a/python/apps/taiga/tests/utils/factories/notifications.py b/python/apps/taiga/tests/utils/factories/notifications.py new file mode 100644 index 000000000..54daa5794 --- /dev/null +++ b/python/apps/taiga/tests/utils/factories/notifications.py @@ -0,0 +1,26 @@ +# -*- coding: utf-8 -*- +# This Source Code Form is subject to the terms of the Mozilla Public +# License, v. 2.0. If a copy of the MPL was not distributed with this +# file, You can obtain one at http://mozilla.org/MPL/2.0/. +# +# Copyright (c) 2023-present Kaleidos INC + +from asgiref.sync import sync_to_async + +from .base import Factory + + +class NotificationFactory(Factory): + type = "test_notification" + + class Meta: + model = "notifications.Notification" + + +@sync_to_async +def create_notification(**kwargs): + return NotificationFactory.create(**kwargs) + + +def build_notification(**kwargs): + return NotificationFactory.build(**kwargs) diff --git a/python/docs/events.md b/python/docs/events.md index 4911d3d17..ed56119d3 100644 --- a/python/docs/events.md +++ b/python/docs/events.md @@ -931,3 +931,16 @@ Content for: "attachment": {... "attachment object" ...} } ``` + + +#### `notifications.create` + +It happens when a new notifications is created. + +Content for: +- user channel: + ``` + { + "notification": {... "notification object" ...}, + } + ``` \ No newline at end of file diff --git a/python/docs/postman/taiga.postman_collection.json b/python/docs/postman/taiga.postman_collection.json index 97108daec..b132f3e74 100644 --- a/python/docs/postman/taiga.postman_collection.json +++ b/python/docs/postman/taiga.postman_collection.json @@ -1,9 +1,9 @@ { "info": { - "_postman_id": "b95931e2-27bf-4229-bdcb-363116c53bea", + "_postman_id": "b330253f-040c-48f5-b6e8-a7d24d0b18f3", "name": "taiga-next", "schema": "https://schema.getpostman.com/json/collection/v2.1.0/collection.json", - "_exporter_id": "9835734" + "_exporter_id": "15018493" }, "item": [ { @@ -3758,6 +3758,109 @@ } ] }, + { + "name": "notifications", + "item": [ + { + "name": "list my notifications", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "" + ], + "type": "text/javascript" + } + } + ], + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "GET", + "header": [], + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/my/notifications", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "my", + "notifications" + ], + "query": [ + { + "key": "read", + "value": "false", + "description": "boolean or none", + "disabled": true + } + ] + } + }, + "response": [] + }, + { + "name": "count my notifications", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "" + ], + "type": "text/javascript" + } + } + ], + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "GET", + "header": [], + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/my/notifications/count", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "my", + "notifications", + "count" + ], + "query": [ + { + "key": "", + "value": null, + "disabled": true + } + ] + } + }, + "response": [] + } + ] + }, { "name": "system", "item": [ diff --git a/python/docs/postman/taiga.postman_collection_e2e.json b/python/docs/postman/taiga.postman_collection_e2e.json index 8133b176a..6f4fec021 100644 --- a/python/docs/postman/taiga.postman_collection_e2e.json +++ b/python/docs/postman/taiga.postman_collection_e2e.json @@ -1,9 +1,9 @@ { "info": { - "_postman_id": "cf393355-0c41-4532-8f51-592b8ef092ab", + "_postman_id": "882b348a-4bc9-4ce2-a3fe-18e77c30fe4f", "name": "taiga-next e2e", "schema": "https://schema.getpostman.com/json/collection/v2.1.0/collection.json", - "_exporter_id": "9835734" + "_exporter_id": "15018493" }, "item": [ { @@ -10025,6 +10025,925 @@ } ] }, + { + "name": "notifications", + "item": [ + { + "name": "N/A log in as 1user", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "pm.test(\"Environment variable settings\", function () {", + " pm.environment.set(\"auth_token\", pm.response.json().token);", + " pm.environment.set(\"refresh_token\", pm.response.json().refresh);", + "});", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "method": "POST", + "header": [ + { + "warning": "This is a duplicate header and will be overridden by the Content-Type header generated by Postman.", + "key": "Content-Type", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"username\": \"1user\",\n \"password\": \"123123\"\n}" + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/auth/token", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "auth", + "token" + ] + } + }, + "response": [] + }, + { + "name": "N/A my.workspaces", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});", + "pm.test(\"Response JSON is correct\", function () {", + " var jsonRes = pm.response.json();", + " ", + " pm.expect(jsonRes).to.be.length.above(0);", + " pm.environment.set(\"ws-id\", jsonRes[0].id);", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "GET", + "header": [], + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/my/workspaces", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "my", + "workspaces" + ] + } + }, + "response": [] + }, + { + "name": "N/A 200 projects", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "pm.test(\"Environment variable settings\", function () {", + " pm.environment.set(\"pj-id\", pm.response.json().id);", + "});", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "", + "});", + "pm.test(\"Response JSON is correct\", function () {", + " var jsonRes = pm.response.json();", + "", + " // Validate response API fields", + " pm.expect(jsonRes).to.have.property(\"logo\");", + " pm.expect(jsonRes).to.have.property(\"logoSmall\");", + " pm.expect(jsonRes).to.have.property(\"logoLarge\");", + " pm.expect(jsonRes).to.have.property(\"id\");", + " pm.expect(jsonRes).to.have.property(\"slug\");", + " pm.expect(jsonRes).to.have.property(\"workspace\");", + " pm.expect(jsonRes).to.have.property(\"workflows\");", + " pm.expect(jsonRes).to.have.property(\"userIsAdmin\");", + " pm.expect(jsonRes).to.have.property(\"userIsMember\");", + " pm.expect(jsonRes).to.have.property(\"userPermissions\");", + " pm.expect(jsonRes).to.have.property(\"userHasPendingInvitation\");", + "", + " pm.expect(jsonRes.description).to.be.eql(\"My description\");", + " pm.expect(jsonRes.name).to.be.eql(\"My project\");", + "", + " var workspace = jsonRes.workspace;", + " pm.expect(workspace).to.have.property(\"id\");", + " pm.expect(workspace).to.have.property(\"name\");", + " pm.expect(workspace).to.have.property(\"userRole\");", + " pm.expect(workspace).to.have.property(\"slug\");", + "", + " pm.expect(workspace.id).to.be.eql(pm.environment.get(\"ws-id\"));", + "", + " var workflows = jsonRes.workflows;", + " pm.expect(workflows[0]).to.have.property(\"name\");", + " pm.expect(workflows[0]).to.have.property(\"slug\");", + "", + " // Validate we're not returning more fields than expected", + " var numOfReturnedFields = Object.keys(workspace).length;", + " pm.expect(numOfReturnedFields).to.equal(4);", + " numOfReturnedFields = Object.keys(jsonRes).length;", + " pm.expect(numOfReturnedFields).to.equal(14);", + "});" + ], + "type": "text/javascript" + } + } + ], + "protocolProfileBehavior": { + "disabledSystemHeaders": {} + }, + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "POST", + "header": [], + "body": { + "mode": "formdata", + "formdata": [ + { + "key": "description", + "value": "My description", + "type": "text" + }, + { + "key": "workspaceId", + "value": "{{ws-id}}", + "type": "text" + }, + { + "key": "logo", + "type": "file", + "src": "/home/daniel/Pictures/phidays_2021.png", + "disabled": true + }, + { + "key": "color", + "value": "2", + "type": "text" + }, + { + "key": "name", + "value": "My project", + "type": "text" + } + ] + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/projects", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "projects" + ] + } + }, + "response": [] + }, + { + "name": "N/A 200 projects.{pj}.invitations", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "POST", + "header": [], + "body": { + "mode": "raw", + "raw": "{\n \"invitations\": [\n {\n \"email\": \"4user@taiga.demo\",\n \"roleSlug\": \"general\"\n },\n {\n \"email\": \"user-test@example.com\",\n \"roleSlug\": \"general\"\n }\n ]\n}", + "options": { + "raw": { + "language": "json" + } + } + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/projects/{{pj-id}}/invitations", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "projects", + "{{pj-id}}", + "invitations" + ] + } + }, + "response": [] + }, + { + "name": "N/A log in as 4user", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "pm.test(\"Environment variable settings\", function () {", + " pm.environment.set(\"auth_token\", pm.response.json().token);", + " pm.environment.set(\"refresh_token\", pm.response.json().refresh);", + "});", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "method": "POST", + "header": [ + { + "warning": "This is a duplicate header and will be overridden by the Content-Type header generated by Postman.", + "key": "Content-Type", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"username\": \"4user\",\n \"password\": \"123123\"\n}" + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/auth/token", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "auth", + "token" + ] + } + }, + "response": [] + }, + { + "name": "N/A projects.{pj}.invitations.accept", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "POST", + "header": [], + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/projects/{{pj-id}}/invitations/accept", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "projects", + "{{pj-id}}", + "invitations", + "accept" + ] + } + }, + "response": [] + }, + { + "name": "N/A log in as 1user", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "pm.test(\"Environment variable settings\", function () {", + " pm.environment.set(\"auth_token\", pm.response.json().token);", + " pm.environment.set(\"refresh_token\", pm.response.json().refresh);", + "});", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "method": "POST", + "header": [ + { + "warning": "This is a duplicate header and will be overridden by the Content-Type header generated by Postman.", + "key": "Content-Type", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"username\": \"1user\",\n \"password\": \"123123\"\n}" + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/auth/token", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "auth", + "token" + ] + } + }, + "response": [] + }, + { + "name": "N/A get workflows and statuses", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "pm.test(\"Environment variable settings\", function () {", + " statuses = pm.response.json()[0].statuses;", + " pm.environment.set(\"wfs-new\", statuses[0].id);", + " pm.environment.set(\"wfs-ready\", statuses[1].id);", + " pm.environment.set(\"wfs-in-progress\", statuses[2].id);", + " pm.environment.set(\"wfs-done\", statuses[3].id);", + "});" + ], + "type": "text/javascript" + } + } + ], + "protocolProfileBehavior": { + "disabledSystemHeaders": {} + }, + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "GET", + "header": [], + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/projects/{{pj-id}}/workflows", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "projects", + "{{pj-id}}", + "workflows" + ] + } + }, + "response": [] + }, + { + "name": "N/A 200.projects.{pj}.workflows.{wf}.stories", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});", + "", + "pm.test(\"Response JSON is correct\", function () {", + " var jsonRes = pm.response.json();", + "", + " pm.environment.set(\"ref2\", jsonRes.ref);", + "", + "});" + ], + "type": "text/javascript" + } + } + ], + "protocolProfileBehavior": { + "disabledSystemHeaders": {} + }, + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "POST", + "header": [], + "body": { + "mode": "raw", + "raw": "{\n \"title\": \"story1\",\n \"status\": \"{{wfs-done}}\"\n}", + "options": { + "raw": { + "language": "json" + } + } + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/projects/{{pj-id}}/workflows/{{wf-slug}}/stories", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "projects", + "{{pj-id}}", + "workflows", + "{{wf-slug}}", + "stories" + ] + } + }, + "response": [] + }, + { + "name": "N/A 200.projects.{pj}.workflows.{wf}.stories", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});", + "", + "pm.test(\"Response JSON is correct\", function () {", + " var jsonRes = pm.response.json();", + "", + " pm.environment.set(\"story_ref\", jsonRes.ref);", + " ", + "});" + ], + "type": "text/javascript" + } + } + ], + "protocolProfileBehavior": { + "disabledSystemHeaders": {} + }, + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "POST", + "header": [], + "body": { + "mode": "raw", + "raw": "{\n \"title\": \"story1\",\n \"status\": \"{{wfs-done}}\"\n}", + "options": { + "raw": { + "language": "json" + } + } + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/projects/{{pj-id}}/workflows/{{wf-slug}}/stories", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "projects", + "{{pj-id}}", + "workflows", + "{{wf-slug}}", + "stories" + ] + } + }, + "response": [] + }, + { + "name": "N/A 200.projects.{pj}.stories.{ref}.assignments", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});", + "", + "pm.test(\"Response JSON is correct\", function () {", + " var jsonRes = pm.response.json();", + " // Validate response API contract", + " pm.expect(jsonRes).to.have.property(\"story\");", + " pm.expect(jsonRes).to.have.property(\"user\");", + " // Validate we're not returning more fields than expected", + " var numOfReturnedFields = Object.keys(jsonRes).length;", + " pm.expect(numOfReturnedFields).to.equal(2);", + "", + " // Validate second-level entries", + " pm.expect(jsonRes.story).to.have.property(\"ref\");", + " pm.expect(jsonRes.story).to.have.property(\"title\");", + " var numOfReturnedFields = Object.keys(jsonRes.story).length;", + " pm.expect(numOfReturnedFields).to.equal(2);", + "", + " pm.expect(jsonRes.user).to.have.property(\"username\");", + " pm.expect(jsonRes.user).to.have.property(\"fullName\");", + " pm.expect(jsonRes.user).to.have.property(\"color\");", + " numOfReturnedFields = Object.keys(jsonRes.user).length;", + " pm.expect(numOfReturnedFields).to.equal(3);", + "", + " pm.environment.set(\"username\", \"1user\");", + "});" + ], + "type": "text/javascript" + } + } + ], + "protocolProfileBehavior": { + "disabledSystemHeaders": {} + }, + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "POST", + "header": [], + "body": { + "mode": "raw", + "raw": "{\n \"username\": \"4user\"\n}", + "options": { + "raw": { + "language": "json" + } + } + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/projects/{{pj-id}}/stories/{{story_ref}}/assignments", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "projects", + "{{pj-id}}", + "stories", + "{{story_ref}}", + "assignments" + ] + } + }, + "response": [] + }, + { + "name": "N/A 200.projects.{pj}.stories.{ref}.assignments.{username}", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(204);", + "});" + ], + "type": "text/javascript" + } + } + ], + "protocolProfileBehavior": { + "disabledSystemHeaders": {} + }, + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "DELETE", + "header": [], + "body": { + "mode": "raw", + "raw": "", + "options": { + "raw": { + "language": "json" + } + } + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/projects/{{pj-id}}/stories/{{story_ref}}/assignments/4user", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "projects", + "{{pj-id}}", + "stories", + "{{story_ref}}", + "assignments", + "4user" + ] + } + }, + "response": [] + }, + { + "name": "N/A log in as 4user", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "pm.test(\"Environment variable settings\", function () {", + " pm.environment.set(\"auth_token\", pm.response.json().token);", + " pm.environment.set(\"refresh_token\", pm.response.json().refresh);", + "});", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "method": "POST", + "header": [ + { + "warning": "This is a duplicate header and will be overridden by the Content-Type header generated by Postman.", + "key": "Content-Type", + "value": "application/json" + } + ], + "body": { + "mode": "raw", + "raw": "{\n \"username\": \"4user\",\n \"password\": \"123123\"\n}" + }, + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/auth/token", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "auth", + "token" + ] + } + }, + "response": [] + }, + { + "name": "200 my.notiffications.count", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "", + "});", + "pm.test(\"Response JSON is correct\", function () {", + " var jsonRes = pm.response.json();", + "", + " // Validate response API fields", + " pm.expect(jsonRes).to.have.property(\"total\");", + " pm.expect(jsonRes).to.have.property(\"read\");", + " pm.expect(jsonRes).to.have.property(\"unread\");", + " // Validate we're not returning more fields than expected", + " var numOfReturnedFields = Object.keys(jsonRes).length;", + " pm.expect(numOfReturnedFields).to.equal(3);", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "GET", + "header": [], + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/my/notifications/count", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "my", + "notifications", + "count" + ] + } + }, + "response": [] + }, + { + "name": "200 my.notiffications", + "event": [ + { + "listen": "test", + "script": { + "exec": [ + "// Post-request execution tasks", + "", + "// Tests", + "pm.test(\"HTTP status code is correct\", function () {", + " pm.response.to.have.status(200);", + "", + "});", + "pm.test(\"Response JSON is correct\", function () {", + " var jsonRes = pm.response.json();", + "", + " // Validate response API contract for each element", + " pm.expect(jsonRes[0]).to.have.property(\"id\");", + " pm.expect(jsonRes[0]).to.have.property(\"type\");", + " pm.expect(jsonRes[0]).to.have.property(\"createdBy\");", + " pm.expect(jsonRes[0]).to.have.property(\"createdAt\");", + " pm.expect(jsonRes[0]).to.have.property(\"readAt\");", + " pm.expect(jsonRes[0]).to.have.property(\"content\");", + " ", + " // Validate we're not returning more fields than expected", + " var numOfReturnedFields = Object.keys(jsonRes[0]).length;", + " pm.expect(numOfReturnedFields).to.equal(6);", + "});" + ], + "type": "text/javascript" + } + } + ], + "request": { + "auth": { + "type": "bearer", + "bearer": [ + { + "key": "token", + "value": "{{auth_token}}", + "type": "string" + } + ] + }, + "method": "GET", + "header": [], + "url": { + "raw": "{{protocol}}://{{domain}}{{port}}{{api_url}}/my/notifications?read=false", + "protocol": "{{protocol}}", + "host": [ + "{{domain}}{{port}}{{api_url}}" + ], + "path": [ + "my", + "notifications" + ], + "query": [ + { + "key": "read", + "value": "false" + } + ] + } + }, + "response": [] + } + ] + }, { "name": "system", "item": [