From 6ac18444faf0cc4c15e9248860c1285ff90879d1 Mon Sep 17 00:00:00 2001 From: Andrew Tavis McAllister Date: Sat, 21 Sep 2024 23:02:51 +0200 Subject: [PATCH 1/8] #938 Exapnd fixtures to product human readable results --- backend/authentication/admin.py | 2 +- backend/authentication/factories.py | 1 + backend/authentication/models.py | 1 + .../management/commands/populate_db.py | 80 +++++++-------- backend/backend/settings.py | 2 +- backend/content/serializers.py | 4 +- backend/entities/factories.py | 12 +-- backend/entities/models.py | 5 +- backend/entities/serializers.py | 3 +- backend/entities/views.py | 14 +-- backend/events/factories.py | 4 +- backend/events/models.py | 3 +- backend/fixtures/superuser.json | 2 +- backend/fixtures/topics.json | 98 +++++++++++++++++++ backend/utils/models.py | 7 ++ backend/utils/utils.py | 30 +----- docker-compose.yml | 3 +- .../components/card/about/CardAboutGroup.vue | 5 +- .../card/search-result/CardSearchResult.vue | 4 +- frontend/composables/fetch.ts | 31 ++++-- .../organizations/[id]/groups/[id]/about.vue | 7 +- frontend/stores/event.ts | 11 ++- frontend/stores/group.ts | 16 ++- frontend/stores/organization.ts | 16 +-- 24 files changed, 236 insertions(+), 125 deletions(-) create mode 100644 backend/fixtures/topics.json create mode 100644 backend/utils/models.py diff --git a/backend/authentication/admin.py b/backend/authentication/admin.py index 9761b2108..ebd476ccb 100644 --- a/backend/authentication/admin.py +++ b/backend/authentication/admin.py @@ -86,7 +86,7 @@ class UserAdmin(BaseUserAdmin): add_form = UserCreationForm # The fields to be used in displaying the User model. - list_display = ["email", "is_admin"] + list_display = ["username", "email", "is_admin"] list_filter = ["is_admin"] fieldsets = [ (None, {"fields": ["email", "password"]}), diff --git a/backend/authentication/factories.py b/backend/authentication/factories.py index e833d3782..987419dc1 100644 --- a/backend/authentication/factories.py +++ b/backend/authentication/factories.py @@ -39,6 +39,7 @@ class Meta: username = factory.Faker("user_name") name = factory.Faker("name") + location = factory.Faker("city") description = factory.Faker("text", max_nb_chars=500) verified = factory.Faker("boolean") verification_method = factory.Faker("word") diff --git a/backend/authentication/models.py b/backend/authentication/models.py index 5f1f3d18b..4ad430c5e 100644 --- a/backend/authentication/models.py +++ b/backend/authentication/models.py @@ -87,6 +87,7 @@ class UserModel(AbstractUser, PermissionsMixin): username = models.CharField(max_length=255, unique=True) name = models.CharField(max_length=255, blank=True) password = models.CharField(max_length=255) + location = models.CharField(max_length=100, blank=True) description = models.TextField(max_length=500, blank=True) verified = models.BooleanField(default=False) verification_method = models.CharField(max_length=30, blank=True) diff --git a/backend/backend/management/commands/populate_db.py b/backend/backend/management/commands/populate_db.py index 262bd4280..09d376ddd 100644 --- a/backend/backend/management/commands/populate_db.py +++ b/backend/backend/management/commands/populate_db.py @@ -1,10 +1,12 @@ +import random from argparse import ArgumentParser from typing import TypedDict, Unpack from django.core.management.base import BaseCommand -from authentication.factories import UserFactory +from authentication.factories import UserFactory, UserTopicFactory from authentication.models import UserModel +from content.models import Topic from entities.factories import ( GroupFactory, GroupTextFactory, @@ -18,9 +20,9 @@ class Options(TypedDict): users: int - orgs: int - groups: int - events: int + orgs_per_user: int + groups_per_org: int + events_per_org: int class Command(BaseCommand): @@ -28,15 +30,15 @@ class Command(BaseCommand): def add_arguments(self, parser: ArgumentParser) -> None: parser.add_argument("--users", type=int, default=10) - parser.add_argument("--opu", type=int, default=1) # orgs per user - parser.add_argument("--gpo", type=int, default=1) # groups per org - parser.add_argument("--epo", type=int, default=1) # events per org + parser.add_argument("--orgs-per-user", type=int, default=1) + parser.add_argument("--groups-per-org", type=int, default=1) + parser.add_argument("--events-per-org", type=int, default=1) def handle(self, *args: str, **options: Unpack[Options]) -> None: - n_users = options.get("users") - n_orgs_per_user = options.get("opu") - n_groups_per_org = options.get("gpo") - n_events_per_org = options.get("epo") + num_users = options.get("users") + num_orgs_per_user = options.get("orgs_per_user") + num_groups_per_org = options.get("groups_per_org") + num_events_per_org = options.get("events_per_org") # Clear all tables before creating new data. UserModel.objects.exclude(username="admin").delete() @@ -44,73 +46,59 @@ def handle(self, *args: str, **options: Unpack[Options]) -> None: Group.objects.all().delete() Event.objects.all().delete() + topics = Topic.objects.all() + try: users = [ UserFactory(username=f"activist_{i}", name=f"Activist {i}") - for i in range(n_users) + for i in range(num_users) ] - for i, user in enumerate(users): - user_location = "Berlin" - user_topic = "Climate" + for u, user in enumerate(users): + user_topic = random.choice(topics) + UserTopicFactory(user_id=user, topic_id=user_topic) - for _ in range(n_orgs_per_user): + for o in range(num_orgs_per_user): user_org = OrganizationFactory( - name=f"{user_location} {user_topic} Organization {i}", + name=f"{user_topic.name} Organization (u: {u} o: {o})", created_by=user, ) - OrganizationTextFactory( - org_id=user_org, - iso="en", - primary=True, - description="This is an org", - get_involved="Get involved!", - donate_prompt="Donate!", - ) + OrganizationTextFactory(org_id=user_org, iso="en", primary=True) - for g in range(n_groups_per_org): + for g in range(num_groups_per_org): user_org_group = GroupFactory( org_id=user_org, - name=f"{user_location} {user_topic} Group {i}-{g}", + name=f"{user_topic.name} Group (u: {u} o: {o} g: {g})", created_by=user, ) GroupTextFactory( - group_id=user_org_group, - iso="en", - primary=True, - description="This is a group", - get_involved="Get involved!", - donate_prompt="Donate!", + group_id=user_org_group, iso="en", primary=True ) - for e in range(n_events_per_org): + for e in range(num_events_per_org): user_org_event = EventFactory( - name=f"{user_location} {user_topic} Event {i}-{e}", + name=f"{user_topic.name} Event (u: {u} o: {o} e: {e})", created_by=user, ) EventTextFactory( - event_id=user_org_event, - iso="en", - primary=True, - description="This is a group", - get_involved="Get involved!", + event_id=user_org_event, iso="en", primary=True ) self.stdout.write( self.style.ERROR( - f"Number of users created: {n_users}\n" - f"Number of organizations created: {n_users * n_orgs_per_user}\n" - f"Number of groups created: {n_users * n_orgs_per_user * n_groups_per_org}\n" - f"Number of events created: {n_users * n_orgs_per_user * n_events_per_org}\n" + f"Number of users created: {num_users}\n" + f"Number of organizations created: {num_users * num_orgs_per_user}\n" + f"Number of groups created: {num_users * num_orgs_per_user * num_groups_per_org}\n" + f"Number of events created: {num_users * num_orgs_per_user * num_events_per_org}\n" ) ) - except Exception as error: + except TypeError as error: self.stdout.write( self.style.ERROR( - f"An error occurred during the creation of dummy data: {error}" + f"A type error occurred during the creation of dummy data: {error}. Make sure to use dashes for populate_db arguments and that they're of the appropriate types." ) ) diff --git a/backend/backend/settings.py b/backend/backend/settings.py index c2f8f7237..e4d73f852 100644 --- a/backend/backend/settings.py +++ b/backend/backend/settings.py @@ -174,7 +174,7 @@ "rest_framework.throttling.AnonRateThrottle", "rest_framework.throttling.UserRateThrottle", ], - "DEFAULT_THROTTLE_RATES": {"anon": "7/min", "user": "10/min"}, + "DEFAULT_THROTTLE_RATES": {"anon": "20/min", "user": "30/min"}, "DEFAULT_SCHEMA_CLASS": "drf_spectacular.openapi.AutoSchema", "DEFAULT_PAGINATION_CLASS": "rest_framework.pagination.PageNumberPagination", "DEFAULT_PAGINATION_ORDERS_OBJECTS": False, diff --git a/backend/content/serializers.py b/backend/content/serializers.py index e8fe3e32d..77ba9550e 100644 --- a/backend/content/serializers.py +++ b/backend/content/serializers.py @@ -117,13 +117,13 @@ class Meta: fields = "__all__" def validate(self, data: Dict[str, Union[str, int]]) -> Dict[str, Union[str, int]]: - if data["active"] is True and data["deprecation_date"] is not None: + if data["active"] is True and data.get("deprecation_date") is not None: raise serializers.ValidationError( _("Active topics cannot have a deprecation date."), code="active_topic_with_deprecation_error", ) - if data["active"] is False and data["deprecation_date"] is None: + if data["active"] is False and data.get("deprecation_date") is None: raise serializers.ValidationError( _("Deprecated topics must have a deprecation date."), code="inactive_topic_no_deprecation_error", diff --git a/backend/entities/factories.py b/backend/entities/factories.py index b3892ba77..418645f72 100644 --- a/backend/entities/factories.py +++ b/backend/entities/factories.py @@ -103,9 +103,9 @@ class Meta: group_id = factory.SubFactory(GroupFactory) iso = factory.Faker("word") primary = factory.Faker("boolean") - description = factory.Faker("text") - get_involved = factory.Faker("text") - donate_prompt = factory.Faker("text") + description = factory.Faker(provider="text", locale="la", max_nb_chars=1000) + get_involved = factory.Faker(provider="text", locale="la") + donate_prompt = factory.Faker(provider="text", locale="la") class GroupTopicFactory(factory.django.DjangoModelFactory): @@ -190,9 +190,9 @@ class Meta: org_id = factory.SubFactory(OrganizationFactory) iso = "en" primary = factory.Faker("boolean") - description = factory.Faker("text") - get_involved = factory.Faker("text") - donate_prompt = factory.Faker("text") + description = factory.Faker(provider="text", locale="la", max_nb_chars=1000) + get_involved = factory.Faker(provider="text", locale="la") + donate_prompt = factory.Faker(provider="text", locale="la") class OrganizationTopicFactory(factory.django.DjangoModelFactory): diff --git a/backend/entities/models.py b/backend/entities/models.py index 4dcd1d000..a3e4de22f 100644 --- a/backend/entities/models.py +++ b/backend/entities/models.py @@ -8,6 +8,7 @@ from django.db import models from authentication import enums +from utils.models import ISO_CHOICES # MARK: Main Tables @@ -129,7 +130,7 @@ def __str__(self) -> str: class GroupText(models.Model): group_id = models.ForeignKey(Group, on_delete=models.CASCADE) - iso = models.CharField(max_length=2) + iso = models.CharField(max_length=2, choices=ISO_CHOICES) primary = models.BooleanField(default=False) description = models.TextField(max_length=500) get_involved = models.TextField(max_length=500, blank=True) @@ -216,7 +217,7 @@ def __str__(self) -> str: class OrganizationText(models.Model): org_id = models.ForeignKey(Organization, on_delete=models.CASCADE) - iso = models.CharField(max_length=2) + iso = models.CharField(max_length=2, choices=ISO_CHOICES) primary = models.BooleanField(default=False) description = models.TextField(max_length=2500) get_involved = models.TextField(max_length=500, blank=True) diff --git a/backend/entities/serializers.py b/backend/entities/serializers.py index afceeb74b..fb8efea4a 100644 --- a/backend/entities/serializers.py +++ b/backend/entities/serializers.py @@ -79,8 +79,8 @@ def validate(self, data: dict[str, Any]) -> dict[str, Any]: raise serializers.ValidationError( "You must accept the terms of service to create an organization." ) - return data + return data def create(self, validated_data: dict[str, Any]) -> Organization: description = validated_data.pop("description", None) @@ -90,6 +90,7 @@ def create(self, validated_data: dict[str, Any]) -> Organization: org_id=org, description=description ) org.org_text = org_text + return org diff --git a/backend/entities/views.py b/backend/entities/views.py index 972b0d5f0..aab425400 100644 --- a/backend/entities/views.py +++ b/backend/entities/views.py @@ -2,7 +2,7 @@ from django.utils import timezone from rest_framework import status, viewsets from rest_framework.authentication import TokenAuthentication -from rest_framework.permissions import IsAuthenticated +from rest_framework.permissions import IsAuthenticatedOrReadOnly from rest_framework.request import Request from rest_framework.response import Response from rest_framework.throttling import AnonRateThrottle, UserRateThrottle @@ -116,18 +116,15 @@ class OrganizationViewSet(viewsets.ModelViewSet[Organization]): serializer_class = OrganizationSerializer pagination_class = CustomPagination throttle_classes = [AnonRateThrottle, UserRateThrottle] - permission_classes = [ - IsAuthenticated, - ] - authentication_classes = [ - TokenAuthentication, - ] + permission_classes = [IsAuthenticatedOrReadOnly] + authentication_classes = [TokenAuthentication] def create(self, request: Request) -> Response: serializer = self.get_serializer(data=request.data) serializer.is_valid(raise_exception=True) org = serializer.save(created_by=request.user) OrganizationApplication.objects.create(org_id=org) + return Response(serializer.data, status=status.HTTP_201_CREATED) def retrieve(self, request: Request, pk: str | None = None) -> Response: @@ -139,6 +136,7 @@ def retrieve(self, request: Request, pk: str | None = None) -> Response: def list(self, request: Request) -> Response: serializer = self.get_serializer(self.get_queryset(), many=True) + return Response(serializer.data, status=status.HTTP_200_OK) def update(self, request: Request, pk: str | None = None) -> Response: @@ -157,6 +155,7 @@ def update(self, request: Request, pk: str | None = None) -> Response: serializer = self.get_serializer(org, data=request.data, partial=True) serializer.is_valid(raise_exception=True) serializer.save() + return Response(serializer.data, status.HTTP_200_OK) def partial_update(self, request: Request, pk: str | None = None) -> Response: @@ -175,6 +174,7 @@ def partial_update(self, request: Request, pk: str | None = None) -> Response: serializer = self.get_serializer(org, data=request.data, partial=True) serializer.is_valid(raise_exception=True) serializer.save() + return Response(serializer.data, status.HTTP_200_OK) def destroy(self, request: Request, pk: str | None = None) -> Response: diff --git a/backend/events/factories.py b/backend/events/factories.py index c2aafa09d..1a8d961f4 100644 --- a/backend/events/factories.py +++ b/backend/events/factories.py @@ -142,8 +142,8 @@ class Meta: event_id = factory.SubFactory(EventFactory) iso = factory.Faker("word") primary = factory.Faker("boolean") - description = factory.Faker("text") - get_involved = factory.Faker("text") + description = factory.Faker(provider="text", locale="la", max_nb_chars=1000) + get_involved = factory.Faker(provider="text", locale="la") class EventTopicFactory(factory.django.DjangoModelFactory): diff --git a/backend/events/models.py b/backend/events/models.py index 936d8685b..208ed0a4b 100644 --- a/backend/events/models.py +++ b/backend/events/models.py @@ -8,6 +8,7 @@ from django.db import models from backend.mixins.models import CreationDeletionMixin +from utils.models import ISO_CHOICES # MARK: Main Tables @@ -130,7 +131,7 @@ def __str__(self) -> str: class EventText(models.Model): event_id = models.ForeignKey(Event, on_delete=models.CASCADE) - iso = models.CharField(max_length=2) + iso = models.CharField(max_length=2, choices=ISO_CHOICES) primary = models.BooleanField() description = models.TextField(max_length=500) get_involved = models.TextField(max_length=500, blank=True) diff --git a/backend/fixtures/superuser.json b/backend/fixtures/superuser.json index 1b22a93a1..1b741b6fd 100644 --- a/backend/fixtures/superuser.json +++ b/backend/fixtures/superuser.json @@ -20,7 +20,7 @@ "email": "admin@activist.org", "is_high_risk": false, "is_active": true, - "is_admin": false, + "is_admin": true, "is_confirmed": true, "groups": [], "user_permissions": [] diff --git a/backend/fixtures/topics.json b/backend/fixtures/topics.json new file mode 100644 index 000000000..00995272c --- /dev/null +++ b/backend/fixtures/topics.json @@ -0,0 +1,98 @@ +[ +{ + "model": "content.topic", + "pk": "24b970c6-5231-403b-9e9d-7de8375193ef", + "fields": { + "name": "Democracy", + "active": true, + "description": "Democracy", + "creation_date": "2024-09-21T20:21:47.121Z", + "last_updated": "2024-09-21T20:21:47.121Z", + "deprecation_date": null + } +}, +{ + "model": "content.topic", + "pk": "2c649964-f64c-4693-be31-0a9fdf981974", + "fields": { + "name": "Women's Rights", + "active": true, + "description": "Women's Rights", + "creation_date": "2024-09-21T20:24:41.462Z", + "last_updated": "2024-09-21T20:24:41.462Z", + "deprecation_date": null + } +}, +{ + "model": "content.topic", + "pk": "2d49b6e2-8916-4543-83f3-9782e8a9680a", + "fields": { + "name": "Animal Rights", + "active": true, + "description": "Animal Rights", + "creation_date": "2024-09-21T20:22:07.547Z", + "last_updated": "2024-09-21T20:22:07.547Z", + "deprecation_date": null + } +}, +{ + "model": "content.topic", + "pk": "b3c6c355-a084-4766-9963-3e54e1af3004", + "fields": { + "name": "Racial Justice", + "active": true, + "description": "Racial Justice", + "creation_date": "2024-09-21T20:24:28.226Z", + "last_updated": "2024-09-21T20:24:28.226Z", + "deprecation_date": null + } +}, +{ + "model": "content.topic", + "pk": "b8d8c9c2-e75b-4d49-aca7-c22cfb1f8b77", + "fields": { + "name": "Housing", + "active": true, + "description": "Housing", + "creation_date": "2024-09-21T20:22:46.519Z", + "last_updated": "2024-09-21T20:22:46.519Z", + "deprecation_date": null + } +}, +{ + "model": "content.topic", + "pk": "d13860c9-4467-4810-a2d5-7826f3769913", + "fields": { + "name": "Education", + "active": true, + "description": "Education", + "creation_date": "2024-09-21T20:22:23.446Z", + "last_updated": "2024-09-21T20:22:23.446Z", + "deprecation_date": null + } +}, +{ + "model": "content.topic", + "pk": "e3715e45-153d-4cce-b957-d68f5a6d4256", + "fields": { + "name": "Climate", + "active": true, + "description": "Climate", + "creation_date": "2024-09-21T20:20:33.450Z", + "last_updated": "2024-09-21T20:20:33.450Z", + "deprecation_date": null + } +}, +{ + "model": "content.topic", + "pk": "ec635e18-a89c-46d3-9a14-8f84be96af9b", + "fields": { + "name": "LGTBQIA+", + "active": true, + "description": "LGTBQIA+", + "creation_date": "2024-09-21T20:23:58.389Z", + "last_updated": "2024-09-21T20:23:58.389Z", + "deprecation_date": null + } +} +] diff --git a/backend/utils/models.py b/backend/utils/models.py new file mode 100644 index 000000000..e21207136 --- /dev/null +++ b/backend/utils/models.py @@ -0,0 +1,7 @@ +ISO_CHOICES = [ + ("de", "de"), + ("en", "en"), + ("es", "es"), + ("fr", "fr"), + ("pt", "pt"), +] diff --git a/backend/utils/utils.py b/backend/utils/utils.py index a73262c96..f757b2145 100644 --- a/backend/utils/utils.py +++ b/backend/utils/utils.py @@ -12,7 +12,7 @@ def validate_creation_and_deletion_dates(data: Any) -> None: code="invalid_creation_date", ) - if data["creation_date"] < data["deletion_date"]: + if data.get("deletion_date") and data.get("deletion_date") < data["creation_date"]: raise serializers.ValidationError( _("The field deletion_date cannot be before creation_date."), code="invalid_date_order", @@ -20,31 +20,11 @@ def validate_creation_and_deletion_dates(data: Any) -> None: def validate_creation_and_deprecation_dates(data: Any) -> None: - if data["deprecation_date"] < data["creation_date"]: + if ( + data.get("deprecation_date") + and data.get("deprecation_date") < data["creation_date"] + ): raise serializers.ValidationError( _("The field deprecation_date cannot be before creation_date."), code="invalid_date_order", ) - - -def validate_flags_number(data: Any) -> None: - if int(data["total_flags"]) < 0: - raise serializers.ValidationError( - _("The field total_flags cannot be negative."), - code="negative_total_flags", - ) - - -def validate_empty(value: Any, field_name: Any) -> None: - if value == "" or value is None: - raise serializers.ValidationError( - _(f"The field {field_name} has to be filled."), code="empty_field" - ) - - -def validate_object_existence(model: Any, object_id: Any) -> None: - if model.objects.filter(id=object_id).exists(): - raise serializers.ValidationError( - _(f"There is no {model.__name__} object with id {object_id}."), - code="inexistent_object", - ) diff --git a/docker-compose.yml b/docker-compose.yml index cb53c28a4..9873891e0 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -11,7 +11,8 @@ services: python manage.py migrate && python manage.py loaddata fixtures/superuser.json && python manage.py loaddata fixtures/status_types.json && - python manage.py populate_db --users 10 --opu 1 --gpo 1 --epo 1 && + python manage.py loaddata fixtures/topics.json && + python manage.py populate_db --users 10 --orgs-per-user 1 --groups-per-org 1 --events-per-org 1 && python manage.py runserver 0.0.0.0:${BACKEND_PORT}" ports: - "${BACKEND_PORT}:${BACKEND_PORT}" diff --git a/frontend/components/card/about/CardAboutGroup.vue b/frontend/components/card/about/CardAboutGroup.vue index 5cb356e88..12e1de7ff 100644 --- a/frontend/components/card/about/CardAboutGroup.vue +++ b/frontend/components/card/about/CardAboutGroup.vue @@ -91,7 +91,10 @@ const props = defineProps<{ const res = await useAsyncData( async () => - await fetchWithToken(`/entities/group_texts?group_id=${props.group.id}`, {}) + await fetchWithOptionalToken( + `/entities/group_texts?group_id=${props.group.id}`, + {} + ) ); const groupTexts = res.data as unknown as GroupText[]; diff --git a/frontend/components/card/search-result/CardSearchResult.vue b/frontend/components/card/search-result/CardSearchResult.vue index 3e7dc83cf..c65ae89d4 100644 --- a/frontend/components/card/search-result/CardSearchResult.vue +++ b/frontend/components/card/search-result/CardSearchResult.vue @@ -210,11 +210,11 @@ -
{{ description }} -
+

diff --git a/frontend/composables/fetch.ts b/frontend/composables/fetch.ts index cb895551a..238be4195 100644 --- a/frontend/composables/fetch.ts +++ b/frontend/composables/fetch.ts @@ -1,15 +1,30 @@ -export const fetchWithToken = async ( +/** + * Returns data given the authentication status of the user. + * @param url Backend URL to make the request to. + * @param data Data to be returned. + * @returns The resulting data from the table. + */ +export const fetchWithOptionalToken = async ( url: string, data: object | {} | undefined ) => { const token = localStorage.getItem("accessToken"); - const res = await $fetch.raw(BASE_BACKEND_URL + url, { - data, - headers: { - Authorization: `Token ${token}`, - }, - }); + if (token) { + const res = await $fetch.raw(BASE_BACKEND_URL + url, { + data, + headers: { + Authorization: `Token ${token}`, + }, + }); - return res._data; + return res._data; + } else { + const res = await $fetch.raw(BASE_BACKEND_URL + url, { + data, + headers: {}, + }); + + return res._data; + } }; diff --git a/frontend/pages/organizations/[id]/groups/[id]/about.vue b/frontend/pages/organizations/[id]/groups/[id]/about.vue index 81dbfc9c3..a8b5c667c 100644 --- a/frontend/pages/organizations/[id]/groups/[id]/about.vue +++ b/frontend/pages/organizations/[id]/groups/[id]/about.vue @@ -97,9 +97,12 @@ const aboveLargeBP = useBreakpoint("lg"); const { id } = useRoute().params; const [resOrg, resOrgTexts] = await Promise.all([ - useAsyncData(async () => await fetchWithToken(`/entities/groups/${id}`, {})), useAsyncData( - async () => await fetchWithToken(`/entities/group_texts?org_id=${id}`, {}) + async () => await fetchWithOptionalToken(`/entities/groups/${id}`, {}) + ), + useAsyncData( + async () => + await fetchWithOptionalToken(`/entities/group_texts?org_id=${id}`, {}) ), ]); diff --git a/frontend/stores/event.ts b/frontend/stores/event.ts index 5ca47ef40..ce384537f 100644 --- a/frontend/stores/event.ts +++ b/frontend/stores/event.ts @@ -49,25 +49,28 @@ export const useEventStore = defineStore("event", { const [resEvent, resEventTexts] = await Promise.all([ useAsyncData( - async () => await fetchWithToken(`/entities/events/${id}`, {}) + async () => await fetchWithOptionalToken(`/entities/events/${id}`, {}) ), // useAsyncData( // async () => - // await fetchWithToken( + // await fetchWithOptionalToken( // `/entities/event_faq?event_id=${id}`, // {} // ) // ), // useAsyncData( // async () => - // await fetchWithToken( + // await fetchWithOptionalToken( // `/entities/event_resources?event_id=${id}`, // {} // ) // ), useAsyncData( async () => - await fetchWithToken(`/entities/event_texts?event_id=${id}`, {}) + await fetchWithOptionalToken( + `/entities/event_texts?event_id=${id}`, + {} + ) ), ]); diff --git a/frontend/stores/group.ts b/frontend/stores/group.ts index 081ba8bfb..74e312dbb 100644 --- a/frontend/stores/group.ts +++ b/frontend/stores/group.ts @@ -34,29 +34,35 @@ export const useGroupStore = defineStore("group", { const [resGroup, resGroupOrg, resGroupTexts] = await Promise.all([ useAsyncData( - async () => await fetchWithToken(`/entities/groups/${id}`, {}) + async () => await fetchWithOptionalToken(`/entities/groups/${id}`, {}) ), useAsyncData( async () => - await fetchWithToken(`/entities/organizations?group_id=${id}`, {}) + await fetchWithOptionalToken( + `/entities/organizations?group_id=${id}`, + {} + ) ), // useAsyncData( // async () => - // await fetchWithToken( + // await fetchWithOptionalToken( // `/entities/group_faq?group_id=${id}`, // {} // ) // ), // useAsyncData( // async () => - // await fetchWithToken( + // await fetchWithOptionalToken( // `/entities/group_resources?group_id=${id}`, // {} // ) // ), useAsyncData( async () => - await fetchWithToken(`/entities/group_texts?group_id=${id}`, {}) + await fetchWithOptionalToken( + `/entities/group_texts?group_id=${id}`, + {} + ) ), ]); diff --git a/frontend/stores/organization.ts b/frontend/stores/organization.ts index 1dbdbd9d9..a7a23a67f 100644 --- a/frontend/stores/organization.ts +++ b/frontend/stores/organization.ts @@ -89,28 +89,29 @@ export const useOrganizationStore = defineStore("organization", { const [responseOrg, responseOrgTexts] = await Promise.all([ useAsyncData( - async () => await fetchWithToken(`/entities/organizations/${id}`, {}) + async () => + await fetchWithOptionalToken(`/entities/organizations/${id}`, {}) ), // useAsyncData( // async () => - // await fetchWithToken( + // await fetchWithOptionalToken( // `/entities/organization_faq?org_id=${id}`, // {} // ) // ), // useAsyncData( - // async () => await fetchWithToken(`/entities/groups?org_id=${id}`, {}) + // async () => await fetchWithOptionalToken(`/entities/groups?org_id=${id}`, {}) // ), // useAsyncData( // async () => - // await fetchWithToken( + // await fetchWithOptionalToken( // `/entities/organization_resources?org_id=${id}`, // {} // ) // ), useAsyncData( async () => - await fetchWithToken( + await fetchWithOptionalToken( `/entities/organization_texts?org_id=${id}`, {} ) @@ -156,7 +157,8 @@ export const useOrganizationStore = defineStore("organization", { const [responseOrgs] = await Promise.all([ useAsyncData( - async () => await fetchWithToken(`/entities/organizations/`, {}) + async () => + await fetchWithOptionalToken(`/entities/organizations/`, {}) ), ]); @@ -167,7 +169,7 @@ export const useOrganizationStore = defineStore("organization", { orgs._value.map((org) => useAsyncData( async () => - await fetchWithToken( + await fetchWithOptionalToken( `/entities/organization_texts?org_id=${org.id}`, {} ) From bab73c247cb17d692f33417ac591b943246d0f19 Mon Sep 17 00:00:00 2001 From: tosta Date: Sun, 22 Sep 2024 21:43:57 +0200 Subject: [PATCH 2/8] fixed mypy errors inside populate_db --- backend/backend/management/commands/populate_db.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/backend/backend/management/commands/populate_db.py b/backend/backend/management/commands/populate_db.py index 09d376ddd..641f0b861 100644 --- a/backend/backend/management/commands/populate_db.py +++ b/backend/backend/management/commands/populate_db.py @@ -35,10 +35,10 @@ def add_arguments(self, parser: ArgumentParser) -> None: parser.add_argument("--events-per-org", type=int, default=1) def handle(self, *args: str, **options: Unpack[Options]) -> None: - num_users = options.get("users") - num_orgs_per_user = options.get("orgs_per_user") - num_groups_per_org = options.get("groups_per_org") - num_events_per_org = options.get("events_per_org") + num_users = options["users"] + num_orgs_per_user = options["orgs_per_user"] + num_groups_per_org = options["groups_per_org"] + num_events_per_org = options["events_per_org"] # Clear all tables before creating new data. UserModel.objects.exclude(username="admin").delete() @@ -64,7 +64,7 @@ def handle(self, *args: str, **options: Unpack[Options]) -> None: created_by=user, ) - OrganizationTextFactory(org_id=user_org, iso="en", primary=True) + OrganizationTextFactory(org_id=user_org, iso="wt", primary=True) for g in range(num_groups_per_org): user_org_group = GroupFactory( From df0438194767f050e6cb33c724f0759137dfd81d Mon Sep 17 00:00:00 2001 From: Andrew Tavis McAllister Date: Mon, 23 Sep 2024 08:59:14 +0200 Subject: [PATCH 3/8] #938 Add dev mode and splash to ease access to orgs and events --- CONTRIBUTING.md | 2 +- README.md | 2 +- frontend/components/FriendlyCaptcha.vue | 5 +- .../card/search-result/CardSearchResult.vue | 2 +- frontend/components/header/HeaderWebsite.vue | 47 ++++++++++++++++++- frontend/components/landing/LandingSplash.vue | 22 +++++++++ frontend/i18n/en-US.json | 4 ++ frontend/stores/dev.ts | 14 ++++++ frontend/stores/modals.ts | 2 +- 9 files changed, 92 insertions(+), 8 deletions(-) create mode 100644 frontend/stores/dev.ts diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 7c977ce57..f239e9c76 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -200,7 +200,7 @@ git remote add upstream https://github.com/activist-org/activist.git # docker compose --env-file .env.dev down ``` -5. You can visit to see the development build once the container is up and running. +5. You can visit to see the development build once the container is up and running. From there click `View organizations` or `View events` to explore the platform. > [!NOTE] > Feel free to contact the team in the [Development room on Matrix](https://matrix.to/#/!CRgLpGeOBNwxYCtqmK:matrix.org?via=matrix.org&via=acter.global&via=chat.0x7cd.xyz) if you're having problems getting your environment setup! diff --git a/README.md b/README.md index d28c5c081..a6ec8fcd5 100644 --- a/README.md +++ b/README.md @@ -210,7 +210,7 @@ git remote add upstream https://github.com/activist-org/activist.git # docker compose --env-file .env.dev down ``` -6. You can then visit to see the development frontend build once the container is up and running. +6. You can then visit to see the development frontend build once the container is up and running. From there click `View organizations` or `View events` to explore the platform. > [!NOTE] > Feel free to contact the team in the [Development room on Matrix](https://matrix.to/#/!CRgLpGeOBNwxYCtqmK:matrix.org?via=matrix.org&via=acter.global&via=chat.0x7cd.xyz) if you're having problems getting your environment setup! If you're having issues with Docker and just want to get the frontend or backend up and running, please see [the section on this in the contributing guide](https://github.com/activist-org/activist/blob/main/CONTRIBUTING.md#using-yarn-or-python). diff --git a/frontend/components/FriendlyCaptcha.vue b/frontend/components/FriendlyCaptcha.vue index 1130246b8..1d528d7bb 100644 --- a/frontend/components/FriendlyCaptcha.vue +++ b/frontend/components/FriendlyCaptcha.vue @@ -7,7 +7,7 @@ }" > { console.log("Captcha response:", response); diff --git a/frontend/components/card/search-result/CardSearchResult.vue b/frontend/components/card/search-result/CardSearchResult.vue index c65ae89d4..73b757805 100644 --- a/frontend/components/card/search-result/CardSearchResult.vue +++ b/frontend/components/card/search-result/CardSearchResult.vue @@ -211,7 +211,7 @@ -->

{{ description }}

diff --git a/frontend/components/header/HeaderWebsite.vue b/frontend/components/header/HeaderWebsite.vue index 928d680c5..46b0a9381 100644 --- a/frontend/components/header/HeaderWebsite.vue +++ b/frontend/components/header/HeaderWebsite.vue @@ -52,7 +52,47 @@ + + + +
+ + diff --git a/frontend/i18n/en-US.json b/frontend/i18n/en-US.json index 098141e2e..59bf2a1b5 100644 --- a/frontend/i18n/en-US.json +++ b/frontend/i18n/en-US.json @@ -248,6 +248,10 @@ "components.landing_splash.message_1": "A platform for growing our movements and organizing actions.", "components.landing_splash.message_2": "Free, open-source, privacy-focused and governed by our community.", "components.landing_splash.request_access_aria_label": "Request access to activist.org", + "components.landing_splash.view_events": "View events", + "components.landing_splash.view_events_aria_label": "View the events section of the activist platform", + "components.landing_splash.view_organizations": "View organizations", + "components.landing_splash.view_organizations_aria_label": "View the organizations section of the activist platform", "components.landing_tech_banner.open_header": "Open", "components.landing_tech_banner.open_source_tagline": "Our code and processes", "components.landing_tech_banner.open_source_text": "We're dedicated to working in the open to build trust with our partner organizations and fellow activists. All who want to help us build are welcome!", diff --git a/frontend/stores/dev.ts b/frontend/stores/dev.ts new file mode 100644 index 000000000..a7ac6d6d8 --- /dev/null +++ b/frontend/stores/dev.ts @@ -0,0 +1,14 @@ +import { useLocalStorage } from "@vueuse/core"; +import { defineStore } from "pinia"; + +export const useDevMode = defineStore("devMode", { + state: () => ({ + active: useLocalStorage("active", false), + }), + + actions: { + check() { + this.active = window.location.href.includes("localhost:3000"); + }, + }, +}); diff --git a/frontend/stores/modals.ts b/frontend/stores/modals.ts index d47fe7f22..d5578ae6d 100644 --- a/frontend/stores/modals.ts +++ b/frontend/stores/modals.ts @@ -11,7 +11,7 @@ export const useModals = defineStore("modals", { actions: { openModal(modalName: string) { - const modals = this.modals; + const { modals } = this; for (const key in modals) { modals[key].isOpen = false; } From 564c60f6c3c0dc1d20b4f12c399bc27fd9f23d69 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 24 Sep 2024 00:08:07 +0000 Subject: [PATCH 4/8] Bump rollup from 4.14.1 to 4.22.4 in /frontend Bumps [rollup](https://github.com/rollup/rollup) from 4.14.1 to 4.22.4. - [Release notes](https://github.com/rollup/rollup/releases) - [Changelog](https://github.com/rollup/rollup/blob/master/CHANGELOG.md) - [Commits](https://github.com/rollup/rollup/compare/v4.14.1...v4.22.4) --- updated-dependencies: - dependency-name: rollup dependency-type: direct:development ... Signed-off-by: dependabot[bot] --- frontend/package.json | 2 +- frontend/yarn.lock | 297 ++++++++++++++---------------------------- 2 files changed, 100 insertions(+), 199 deletions(-) diff --git a/frontend/package.json b/frontend/package.json index f1aab1940..e5d6449f7 100644 --- a/frontend/package.json +++ b/frontend/package.json @@ -35,7 +35,7 @@ "playwright": "1.45.1", "prettier": "3.2.5", "prettier-plugin-tailwindcss": "0.5.13", - "rollup": "4.14.1", + "rollup": "4.22.4", "typescript": "5.4.4", "vite": "5.2.14", "vue": "3.4.21", diff --git a/frontend/yarn.lock b/frontend/yarn.lock index 2d5021427..74ba214bd 100644 --- a/frontend/yarn.lock +++ b/frontend/yarn.lock @@ -1913,160 +1913,85 @@ estree-walker "^2.0.2" picomatch "^2.3.1" -"@rollup/rollup-android-arm-eabi@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.14.1.tgz#ca0501dd836894216cb9572848c5dde4bfca3bec" - integrity sha512-fH8/o8nSUek8ceQnT7K4EQbSiV7jgkHq81m9lWZFIXjJ7lJzpWXbQFpT/Zh6OZYnpFykvzC3fbEvEAFZu03dPA== - -"@rollup/rollup-android-arm-eabi@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.21.0.tgz#d941173f82f9b041c61b0dc1a2a91dcd06e4b31e" - integrity sha512-WTWD8PfoSAJ+qL87lE7votj3syLavxunWhzCnx3XFxFiI/BA/r3X7MUM8dVrH8rb2r4AiO8jJsr3ZjdaftmnfA== - -"@rollup/rollup-android-arm64@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.14.1.tgz#154ca7e4f815d2e442ffc62ee7f64aee8b2547b0" - integrity sha512-Y/9OHLjzkunF+KGEoJr3heiD5X9OLa8sbT1lm0NYeKyaM3oMhhQFvPB0bNZYJwlq93j8Z6wSxh9+cyKQaxS7PQ== - -"@rollup/rollup-android-arm64@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.21.0.tgz#7e7157c8543215245ceffc445134d9e843ba51c0" - integrity sha512-a1sR2zSK1B4eYkiZu17ZUZhmUQcKjk2/j9Me2IDjk1GHW7LB5Z35LEzj9iJch6gtUfsnvZs1ZNyDW2oZSThrkA== - -"@rollup/rollup-darwin-arm64@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.14.1.tgz#02b522ab6ccc2c504634651985ff8e657b42c055" - integrity sha512-+kecg3FY84WadgcuSVm6llrABOdQAEbNdnpi5X3UwWiFVhZIZvKgGrF7kmLguvxHNQy+UuRV66cLVl3S+Rkt+Q== - -"@rollup/rollup-darwin-arm64@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.21.0.tgz#f0a18a4fc8dc6eb1e94a51fa2adb22876f477947" - integrity sha512-zOnKWLgDld/svhKO5PD9ozmL6roy5OQ5T4ThvdYZLpiOhEGY+dp2NwUmxK0Ld91LrbjrvtNAE0ERBwjqhZTRAA== - -"@rollup/rollup-darwin-x64@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.14.1.tgz#217737f9f73de729fdfd7d529afebb6c8283f554" - integrity sha512-2pYRzEjVqq2TB/UNv47BV/8vQiXkFGVmPFwJb+1E0IFFZbIX8/jo1olxqqMbo6xCXf8kabANhp5bzCij2tFLUA== - -"@rollup/rollup-darwin-x64@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.21.0.tgz#34b7867613e5cc42d2b85ddc0424228cc33b43f0" - integrity sha512-7doS8br0xAkg48SKE2QNtMSFPFUlRdw9+votl27MvT46vo44ATBmdZdGysOevNELmZlfd+NEa0UYOA8f01WSrg== - -"@rollup/rollup-linux-arm-gnueabihf@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.14.1.tgz#a87e478ab3f697c7f4e74c8b1cac1e0667f8f4be" - integrity sha512-mS6wQ6Do6/wmrF9aTFVpIJ3/IDXhg1EZcQFYHZLHqw6AzMBjTHWnCG35HxSqUNphh0EHqSM6wRTT8HsL1C0x5g== - -"@rollup/rollup-linux-arm-gnueabihf@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.21.0.tgz#422b19ff9ae02b05d3395183d1d43b38c7c8be0b" - integrity sha512-pWJsfQjNWNGsoCq53KjMtwdJDmh/6NubwQcz52aEwLEuvx08bzcy6tOUuawAOncPnxz/3siRtd8hiQ32G1y8VA== - -"@rollup/rollup-linux-arm-musleabihf@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.21.0.tgz#568aa29195ef6fc57ec6ed3f518923764406a8ee" - integrity sha512-efRIANsz3UHZrnZXuEvxS9LoCOWMGD1rweciD6uJQIx2myN3a8Im1FafZBzh7zk1RJ6oKcR16dU3UPldaKd83w== - -"@rollup/rollup-linux-arm64-gnu@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.14.1.tgz#4da6830eca27e5f4ca15f9197e5660952ca185c6" - integrity sha512-p9rGKYkHdFMzhckOTFubfxgyIO1vw//7IIjBBRVzyZebWlzRLeNhqxuSaZ7kCEKVkm/kuC9fVRW9HkC/zNRG2w== - -"@rollup/rollup-linux-arm64-gnu@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.21.0.tgz#22309c8bcba9a73114f69165c72bc94b2fbec085" - integrity sha512-ZrPhydkTVhyeGTW94WJ8pnl1uroqVHM3j3hjdquwAcWnmivjAwOYjTEAuEDeJvGX7xv3Z9GAvrBkEzCgHq9U1w== - -"@rollup/rollup-linux-arm64-musl@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.14.1.tgz#0b0ed35720aebc8f5e501d370a9ea0f686ead1e0" - integrity sha512-nDY6Yz5xS/Y4M2i9JLQd3Rofh5OR8Bn8qe3Mv/qCVpHFlwtZSBYSPaU4mrGazWkXrdQ98GB//H0BirGR/SKFSw== - -"@rollup/rollup-linux-arm64-musl@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.21.0.tgz#c93c388af6d33f082894b8a60839d7265b2b9bc5" - integrity sha512-cfaupqd+UEFeURmqNP2eEvXqgbSox/LHOyN9/d2pSdV8xTrjdg3NgOFJCtc1vQ/jEke1qD0IejbBfxleBPHnPw== - -"@rollup/rollup-linux-powerpc64le-gnu@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.14.1.tgz#024ad04d162726f25e62915851f7df69a9677c17" - integrity sha512-im7HE4VBL+aDswvcmfx88Mp1soqL9OBsdDBU8NqDEYtkri0qV0THhQsvZtZeNNlLeCUQ16PZyv7cqutjDF35qw== - -"@rollup/rollup-linux-powerpc64le-gnu@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.21.0.tgz#493c5e19e395cf3c6bd860c7139c8a903dea72b4" - integrity sha512-ZKPan1/RvAhrUylwBXC9t7B2hXdpb/ufeu22pG2psV7RN8roOfGurEghw1ySmX/CmDDHNTDDjY3lo9hRlgtaHg== - -"@rollup/rollup-linux-riscv64-gnu@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.14.1.tgz#180694d1cd069ddbe22022bb5b1bead3b7de581c" - integrity sha512-RWdiHuAxWmzPJgaHJdpvUUlDz8sdQz4P2uv367T2JocdDa98iRw2UjIJ4QxSyt077mXZT2X6pKfT2iYtVEvOFw== - -"@rollup/rollup-linux-riscv64-gnu@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.21.0.tgz#a2eab4346fbe5909165ce99adb935ba30c9fb444" - integrity sha512-H1eRaCwd5E8eS8leiS+o/NqMdljkcb1d6r2h4fKSsCXQilLKArq6WS7XBLDu80Yz+nMqHVFDquwcVrQmGr28rg== - -"@rollup/rollup-linux-s390x-gnu@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.14.1.tgz#f7b4e2b0ca49be4e34f9ef0b548c926d94edee87" - integrity sha512-VMgaGQ5zRX6ZqV/fas65/sUGc9cPmsntq2FiGmayW9KMNfWVG/j0BAqImvU4KTeOOgYSf1F+k6at1UfNONuNjA== - -"@rollup/rollup-linux-s390x-gnu@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.21.0.tgz#0bc49a79db4345d78d757bb1b05e73a1b42fa5c3" - integrity sha512-zJ4hA+3b5tu8u7L58CCSI0A9N1vkfwPhWd/puGXwtZlsB5bTkwDNW/+JCU84+3QYmKpLi+XvHdmrlwUwDA6kqw== - -"@rollup/rollup-linux-x64-gnu@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.14.1.tgz#0aaf79e5b9ccf7db3084fe6c3f2d2873a27d5af4" - integrity sha512-9Q7DGjZN+hTdJomaQ3Iub4m6VPu1r94bmK2z3UeWP3dGUecRC54tmVu9vKHTm1bOt3ASoYtEz6JSRLFzrysKlA== - -"@rollup/rollup-linux-x64-gnu@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.21.0.tgz#4fd36a6a41f3406d8693321b13d4f9b7658dd4b9" - integrity sha512-e2hrvElFIh6kW/UNBQK/kzqMNY5mO+67YtEh9OA65RM5IJXYTWiXjX6fjIiPaqOkBthYF1EqgiZ6OXKcQsM0hg== - -"@rollup/rollup-linux-x64-musl@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.14.1.tgz#38f0a37ca5015eb07dff86a1b6f94279c179f4ed" - integrity sha512-JNEG/Ti55413SsreTguSx0LOVKX902OfXIKVg+TCXO6Gjans/k9O6ww9q3oLGjNDaTLxM+IHFMeXy/0RXL5R/g== - -"@rollup/rollup-linux-x64-musl@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.21.0.tgz#10ebb13bd4469cbad1a5d9b073bd27ec8a886200" - integrity sha512-1vvmgDdUSebVGXWX2lIcgRebqfQSff0hMEkLJyakQ9JQUbLDkEaMsPTLOmyccyC6IJ/l3FZuJbmrBw/u0A0uCQ== - -"@rollup/rollup-win32-arm64-msvc@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.14.1.tgz#84d48c55740ede42c77373f76e85f368633a0cc3" - integrity sha512-ryS22I9y0mumlLNwDFYZRDFLwWh3aKaC72CWjFcFvxK0U6v/mOkM5Up1bTbCRAhv3kEIwW2ajROegCIQViUCeA== - -"@rollup/rollup-win32-arm64-msvc@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.21.0.tgz#2fef1a90f1402258ef915ae5a94cc91a5a1d5bfc" - integrity sha512-s5oFkZ/hFcrlAyBTONFY1TWndfyre1wOMwU+6KCpm/iatybvrRgmZVM+vCFwxmC5ZhdlgfE0N4XorsDpi7/4XQ== - -"@rollup/rollup-win32-ia32-msvc@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.14.1.tgz#c1e0bc39e20e760f0a526ddf14ae0543af796605" - integrity sha512-TdloItiGk+T0mTxKx7Hp279xy30LspMso+GzQvV2maYePMAWdmrzqSNZhUpPj3CGw12aGj57I026PgLCTu8CGg== - -"@rollup/rollup-win32-ia32-msvc@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.21.0.tgz#a18ad47a95c5f264defb60acdd8c27569f816fc1" - integrity sha512-G9+TEqRnAA6nbpqyUqgTiopmnfgnMkR3kMukFBDsiyy23LZvUCpiUwjTRx6ezYCjJODXrh52rBR9oXvm+Fp5wg== - -"@rollup/rollup-win32-x64-msvc@4.14.1": - version "4.14.1" - resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.14.1.tgz#299eee74b7d87e116083ac5b1ce8dd9434668294" - integrity sha512-wQGI+LY/Py20zdUPq+XCem7JcPOyzIJBm3dli+56DJsQOHbnXZFEwgmnC6el1TPAfC8lBT3m+z69RmLykNUbew== - -"@rollup/rollup-win32-x64-msvc@4.21.0": - version "4.21.0" - resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.21.0.tgz#20c09cf44dcb082140cc7f439dd679fe4bba3375" - integrity sha512-2jsCDZwtQvRhejHLfZ1JY6w6kEuEtfF9nzYsZxzSlNVKDX+DpsDJ+Rbjkm74nvg2rdx0gwBS+IMdvwJuq3S9pQ== +"@rollup/rollup-android-arm-eabi@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.22.4.tgz#8b613b9725e8f9479d142970b106b6ae878610d5" + integrity sha512-Fxamp4aEZnfPOcGA8KSNEohV8hX7zVHOemC8jVBoBUHu5zpJK/Eu3uJwt6BMgy9fkvzxDaurgj96F/NiLukF2w== + +"@rollup/rollup-android-arm64@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.22.4.tgz#654ca1049189132ff602bfcf8df14c18da1f15fb" + integrity sha512-VXoK5UMrgECLYaMuGuVTOx5kcuap1Jm8g/M83RnCHBKOqvPPmROFJGQaZhGccnsFtfXQ3XYa4/jMCJvZnbJBdA== + +"@rollup/rollup-darwin-arm64@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.22.4.tgz#6d241d099d1518ef0c2205d96b3fa52e0fe1954b" + integrity sha512-xMM9ORBqu81jyMKCDP+SZDhnX2QEVQzTcC6G18KlTQEzWK8r/oNZtKuZaCcHhnsa6fEeOBionoyl5JsAbE/36Q== + +"@rollup/rollup-darwin-x64@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.22.4.tgz#42bd19d292a57ee11734c980c4650de26b457791" + integrity sha512-aJJyYKQwbHuhTUrjWjxEvGnNNBCnmpHDvrb8JFDbeSH3m2XdHcxDd3jthAzvmoI8w/kSjd2y0udT+4okADsZIw== + +"@rollup/rollup-linux-arm-gnueabihf@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.22.4.tgz#f23555ee3d8fe941c5c5fd458cd22b65eb1c2232" + integrity sha512-j63YtCIRAzbO+gC2L9dWXRh5BFetsv0j0va0Wi9epXDgU/XUi5dJKo4USTttVyK7fGw2nPWK0PbAvyliz50SCQ== + +"@rollup/rollup-linux-arm-musleabihf@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.22.4.tgz#f3bbd1ae2420f5539d40ac1fde2b38da67779baa" + integrity sha512-dJnWUgwWBX1YBRsuKKMOlXCzh2Wu1mlHzv20TpqEsfdZLb3WoJW2kIEsGwLkroYf24IrPAvOT/ZQ2OYMV6vlrg== + +"@rollup/rollup-linux-arm64-gnu@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.22.4.tgz#7abe900120113e08a1f90afb84c7c28774054d15" + integrity sha512-AdPRoNi3NKVLolCN/Sp4F4N1d98c4SBnHMKoLuiG6RXgoZ4sllseuGioszumnPGmPM2O7qaAX/IJdeDU8f26Aw== + +"@rollup/rollup-linux-arm64-musl@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.22.4.tgz#9e655285c8175cd44f57d6a1e8e5dedfbba1d820" + integrity sha512-Gl0AxBtDg8uoAn5CCqQDMqAx22Wx22pjDOjBdmG0VIWX3qUBHzYmOKh8KXHL4UpogfJ14G4wk16EQogF+v8hmA== + +"@rollup/rollup-linux-powerpc64le-gnu@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.22.4.tgz#9a79ae6c9e9d8fe83d49e2712ecf4302db5bef5e" + integrity sha512-3aVCK9xfWW1oGQpTsYJJPF6bfpWfhbRnhdlyhak2ZiyFLDaayz0EP5j9V1RVLAAxlmWKTDfS9wyRyY3hvhPoOg== + +"@rollup/rollup-linux-riscv64-gnu@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.22.4.tgz#67ac70eca4ace8e2942fabca95164e8874ab8128" + integrity sha512-ePYIir6VYnhgv2C5Xe9u+ico4t8sZWXschR6fMgoPUK31yQu7hTEJb7bCqivHECwIClJfKgE7zYsh1qTP3WHUA== + +"@rollup/rollup-linux-s390x-gnu@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.22.4.tgz#9f883a7440f51a22ed7f99e1d070bd84ea5005fc" + integrity sha512-GqFJ9wLlbB9daxhVlrTe61vJtEY99/xB3C8e4ULVsVfflcpmR6c8UZXjtkMA6FhNONhj2eA5Tk9uAVw5orEs4Q== + +"@rollup/rollup-linux-x64-gnu@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.22.4.tgz#70116ae6c577fe367f58559e2cffb5641a1dd9d0" + integrity sha512-87v0ol2sH9GE3cLQLNEy0K/R0pz1nvg76o8M5nhMR0+Q+BBGLnb35P0fVz4CQxHYXaAOhE8HhlkaZfsdUOlHwg== + +"@rollup/rollup-linux-x64-musl@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.22.4.tgz#f473f88219feb07b0b98b53a7923be716d1d182f" + integrity sha512-UV6FZMUgePDZrFjrNGIWzDo/vABebuXBhJEqrHxrGiU6HikPy0Z3LfdtciIttEUQfuDdCn8fqh7wiFJjCNwO+g== + +"@rollup/rollup-win32-arm64-msvc@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.22.4.tgz#4349482d17f5d1c58604d1c8900540d676f420e0" + integrity sha512-BjI+NVVEGAXjGWYHz/vv0pBqfGoUH0IGZ0cICTn7kB9PyjrATSkX+8WkguNjWoj2qSr1im/+tTGRaY+4/PdcQw== + +"@rollup/rollup-win32-ia32-msvc@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.22.4.tgz#a6fc39a15db618040ec3c2a24c1e26cb5f4d7422" + integrity sha512-SiWG/1TuUdPvYmzmYnmd3IEifzR61Tragkbx9D3+R8mzQqDBz8v+BvZNDlkiTtI9T15KYZhP0ehn3Dld4n9J5g== + +"@rollup/rollup-win32-x64-msvc@4.22.4": + version "4.22.4" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.22.4.tgz#3dd5d53e900df2a40841882c02e56f866c04d202" + integrity sha512-j8pPKp53/lq9lMXN57S8cFz0MynJk8OWNuUnXct/9KCpKU7DgU3bYMJhwWmcqC0UU29p8Lr0/7KEVcaM6bf47Q== "@rushstack/eslint-patch@^1.10.3": version "1.10.4" @@ -8694,53 +8619,29 @@ rollup-plugin-visualizer@^5.12.0: source-map "^0.7.4" yargs "^17.5.1" -rollup@4.14.1: - version "4.14.1" - resolved "https://registry.yarnpkg.com/rollup/-/rollup-4.14.1.tgz#228d5159c3f4d8745bd24819d734bc6c6ca87c09" - integrity sha512-4LnHSdd3QK2pa1J6dFbfm1HN0D7vSK/ZuZTsdyUAlA6Rr1yTouUTL13HaDOGJVgby461AhrNGBS7sCGXXtT+SA== - dependencies: - "@types/estree" "1.0.5" - optionalDependencies: - "@rollup/rollup-android-arm-eabi" "4.14.1" - "@rollup/rollup-android-arm64" "4.14.1" - "@rollup/rollup-darwin-arm64" "4.14.1" - "@rollup/rollup-darwin-x64" "4.14.1" - "@rollup/rollup-linux-arm-gnueabihf" "4.14.1" - "@rollup/rollup-linux-arm64-gnu" "4.14.1" - "@rollup/rollup-linux-arm64-musl" "4.14.1" - "@rollup/rollup-linux-powerpc64le-gnu" "4.14.1" - "@rollup/rollup-linux-riscv64-gnu" "4.14.1" - "@rollup/rollup-linux-s390x-gnu" "4.14.1" - "@rollup/rollup-linux-x64-gnu" "4.14.1" - "@rollup/rollup-linux-x64-musl" "4.14.1" - "@rollup/rollup-win32-arm64-msvc" "4.14.1" - "@rollup/rollup-win32-ia32-msvc" "4.14.1" - "@rollup/rollup-win32-x64-msvc" "4.14.1" - fsevents "~2.3.2" - -rollup@^4.13.0, rollup@^4.18.0, rollup@^4.20.0: - version "4.21.0" - resolved "https://registry.yarnpkg.com/rollup/-/rollup-4.21.0.tgz#28db5f5c556a5180361d35009979ccc749560b9d" - integrity sha512-vo+S/lfA2lMS7rZ2Qoubi6I5hwZwzXeUIctILZLbHI+laNtvhhOIon2S1JksA5UEDQ7l3vberd0fxK44lTYjbQ== +rollup@4.22.4, rollup@^4.13.0, rollup@^4.18.0, rollup@^4.20.0: + version "4.22.4" + resolved "https://registry.yarnpkg.com/rollup/-/rollup-4.22.4.tgz#4135a6446671cd2a2453e1ad42a45d5973ec3a0f" + integrity sha512-vD8HJ5raRcWOyymsR6Z3o6+RzfEPCnVLMFJ6vRslO1jt4LO6dUo5Qnpg7y4RkZFM2DMe3WUirkI5c16onjrc6A== dependencies: "@types/estree" "1.0.5" optionalDependencies: - "@rollup/rollup-android-arm-eabi" "4.21.0" - "@rollup/rollup-android-arm64" "4.21.0" - "@rollup/rollup-darwin-arm64" "4.21.0" - "@rollup/rollup-darwin-x64" "4.21.0" - "@rollup/rollup-linux-arm-gnueabihf" "4.21.0" - "@rollup/rollup-linux-arm-musleabihf" "4.21.0" - "@rollup/rollup-linux-arm64-gnu" "4.21.0" - "@rollup/rollup-linux-arm64-musl" "4.21.0" - "@rollup/rollup-linux-powerpc64le-gnu" "4.21.0" - "@rollup/rollup-linux-riscv64-gnu" "4.21.0" - "@rollup/rollup-linux-s390x-gnu" "4.21.0" - "@rollup/rollup-linux-x64-gnu" "4.21.0" - "@rollup/rollup-linux-x64-musl" "4.21.0" - "@rollup/rollup-win32-arm64-msvc" "4.21.0" - "@rollup/rollup-win32-ia32-msvc" "4.21.0" - "@rollup/rollup-win32-x64-msvc" "4.21.0" + "@rollup/rollup-android-arm-eabi" "4.22.4" + "@rollup/rollup-android-arm64" "4.22.4" + "@rollup/rollup-darwin-arm64" "4.22.4" + "@rollup/rollup-darwin-x64" "4.22.4" + "@rollup/rollup-linux-arm-gnueabihf" "4.22.4" + "@rollup/rollup-linux-arm-musleabihf" "4.22.4" + "@rollup/rollup-linux-arm64-gnu" "4.22.4" + "@rollup/rollup-linux-arm64-musl" "4.22.4" + "@rollup/rollup-linux-powerpc64le-gnu" "4.22.4" + "@rollup/rollup-linux-riscv64-gnu" "4.22.4" + "@rollup/rollup-linux-s390x-gnu" "4.22.4" + "@rollup/rollup-linux-x64-gnu" "4.22.4" + "@rollup/rollup-linux-x64-musl" "4.22.4" + "@rollup/rollup-win32-arm64-msvc" "4.22.4" + "@rollup/rollup-win32-ia32-msvc" "4.22.4" + "@rollup/rollup-win32-x64-msvc" "4.22.4" fsevents "~2.3.2" run-applescript@^7.0.0: From 0c0aae51052e6e3a6a1f57fffb2d8608b86a9c86 Mon Sep 17 00:00:00 2001 From: Andrew Tavis McAllister Date: Wed, 25 Sep 2024 01:13:22 +0200 Subject: [PATCH 5/8] Expand event models+ to allow for list and create store --- .../management/commands/populate_db.py | 7 +- backend/backend/settings.py | 2 +- backend/entities/serializers.py | 1 + backend/entities/views.py | 7 ++ backend/events/factories.py | 4 +- backend/events/models.py | 4 + backend/events/serializers.py | 53 ++++++++++-- backend/events/views.py | 76 +++++++++++++++- frontend/assets/css/tailwind.css | 4 + .../components/landing/LandingContent.vue | 4 +- frontend/components/landing/LandingSplash.vue | 38 ++++---- .../components/page/PageCommunityFooter.vue | 4 +- frontend/pages/events/index.vue | 17 ++-- frontend/stores/event.ts | 86 +++++++++++++++++-- frontend/stores/organization.ts | 11 ++- frontend/types/entities/organization.d.ts | 2 +- frontend/types/events/event.d.ts | 45 +++++++++- frontend/utils/testEntities.ts | 6 +- 18 files changed, 306 insertions(+), 65 deletions(-) diff --git a/backend/backend/management/commands/populate_db.py b/backend/backend/management/commands/populate_db.py index 641f0b861..00d373e0e 100644 --- a/backend/backend/management/commands/populate_db.py +++ b/backend/backend/management/commands/populate_db.py @@ -60,7 +60,7 @@ def handle(self, *args: str, **options: Unpack[Options]) -> None: for o in range(num_orgs_per_user): user_org = OrganizationFactory( - name=f"{user_topic.name} Organization (u: {u} o: {o})", + name=f"{user_topic.name} Organization (U{u}-O{o})", created_by=user, ) @@ -69,7 +69,7 @@ def handle(self, *args: str, **options: Unpack[Options]) -> None: for g in range(num_groups_per_org): user_org_group = GroupFactory( org_id=user_org, - name=f"{user_topic.name} Group (u: {u} o: {o} g: {g})", + name=f"{user_topic.name} Group (U{u}-O{o}-G{g})", created_by=user, ) @@ -79,7 +79,8 @@ def handle(self, *args: str, **options: Unpack[Options]) -> None: for e in range(num_events_per_org): user_org_event = EventFactory( - name=f"{user_topic.name} Event (u: {u} o: {o} e: {e})", + name=f"{user_topic.name} Event (U{u}-O{o}-E{e})", + type=random.choice(["learn", "action"]), created_by=user, ) diff --git a/backend/backend/settings.py b/backend/backend/settings.py index e4d73f852..94c45627a 100644 --- a/backend/backend/settings.py +++ b/backend/backend/settings.py @@ -174,7 +174,7 @@ "rest_framework.throttling.AnonRateThrottle", "rest_framework.throttling.UserRateThrottle", ], - "DEFAULT_THROTTLE_RATES": {"anon": "20/min", "user": "30/min"}, + "DEFAULT_THROTTLE_RATES": {"anon": "40/min", "user": "60/min"}, "DEFAULT_SCHEMA_CLASS": "drf_spectacular.openapi.AutoSchema", "DEFAULT_PAGINATION_CLASS": "rest_framework.pagination.PageNumberPagination", "DEFAULT_PAGINATION_ORDERS_OBJECTS": False, diff --git a/backend/entities/serializers.py b/backend/entities/serializers.py index fb8efea4a..7f2ba3097 100644 --- a/backend/entities/serializers.py +++ b/backend/entities/serializers.py @@ -85,6 +85,7 @@ def validate(self, data: dict[str, Any]) -> dict[str, Any]: def create(self, validated_data: dict[str, Any]) -> Organization: description = validated_data.pop("description", None) org = Organization.objects.create(**validated_data) + if org and description: org_text = OrganizationText.objects.create( org_id=org, description=description diff --git a/backend/entities/views.py b/backend/entities/views.py index aab425400..5dec122df 100644 --- a/backend/entities/views.py +++ b/backend/entities/views.py @@ -60,6 +60,7 @@ class GroupViewSet(viewsets.ModelViewSet[Group]): def list(self, request: Request, *args: str, **kwargs: int) -> Response: serializer = self.get_serializer(self.queryset, many=True) + return Response(serializer.data, status=status.HTTP_200_OK) def create(self, request: Request) -> Response: @@ -67,6 +68,7 @@ def create(self, request: Request) -> Response: serializer.is_valid(raise_exception=True) serializer.save(created_by=request.user) data = {"message": f"New Group created: {serializer.data}"} + return Response(data, status=status.HTTP_201_CREATED) def retrieve(self, request: Request, *args: str, **kwargs: int) -> Response: @@ -82,6 +84,7 @@ def partial_update(self, request: Request, *args: str, **kwargs: int) -> Respons return Response( {"error": "Group not found"}, status=status.HTTP_404_NOT_FOUND ) + if request.user != group.created_by: return Response( {"error": "You are not authorized to update this group"}, @@ -91,6 +94,7 @@ def partial_update(self, request: Request, *args: str, **kwargs: int) -> Respons serializer = self.get_serializer(group, data=request.data, partial=True) serializer.is_valid(raise_exception=True) serializer.save() + return Response(serializer.data, status=status.HTTP_200_OK) def destroy(self, request: Request, *args: str, **kwargs: int) -> Response: @@ -100,11 +104,13 @@ def destroy(self, request: Request, *args: str, **kwargs: int) -> Response: return Response( {"error": "Group not found"}, status=status.HTTP_404_NOT_FOUND ) + if request.user != group.created_by: return Response( {"error": "You are not authorized to delete this group"}, status.HTTP_401_UNAUTHORIZED, ) + group.delete() return Response( {"message": "Group deleted successfully"}, status=status.HTTP_200_OK @@ -130,6 +136,7 @@ def create(self, request: Request) -> Response: def retrieve(self, request: Request, pk: str | None = None) -> Response: if org := self.queryset.filter(id=pk).first(): serializer = self.get_serializer(org) + return Response(serializer.data, status=status.HTTP_200_OK) return Response({"error": "Organization not found"}, status.HTTP_404_NOT_FOUND) diff --git a/backend/events/factories.py b/backend/events/factories.py index 1a8d961f4..f89b20e9f 100644 --- a/backend/events/factories.py +++ b/backend/events/factories.py @@ -23,11 +23,13 @@ class EventFactory(factory.django.DjangoModelFactory): class Meta: model = Event + django_get_or_create = ("created_by",) name = factory.Faker("word") tagline = factory.Faker("word") - type = factory.Faker("word") + type = random.choice(["learn", "action"]) online_location_link = factory.Faker("url") + offline_location = factory.Faker("city") offline_location_lat = factory.Faker("latitude") offline_location_long = factory.Faker("longitude") start_time = factory.LazyFunction( diff --git a/backend/events/models.py b/backend/events/models.py index 208ed0a4b..5a9ea051b 100644 --- a/backend/events/models.py +++ b/backend/events/models.py @@ -27,6 +27,7 @@ class Event(CreationDeletionMixin): ) type = models.CharField(max_length=255) online_location_link = models.CharField(max_length=255, blank=True) + offline_location = models.CharField(max_length=255, blank=True) offline_location_lat = models.FloatField(null=True, blank=True) offline_location_long = models.FloatField(null=True, blank=True) get_involved_url = models.URLField(blank=True) @@ -36,6 +37,9 @@ class Event(CreationDeletionMixin): is_private = models.BooleanField(default=False) start_time = models.DateTimeField() end_time = models.DateTimeField() + event_text = models.ForeignKey( + "EventText", on_delete=models.CASCADE, null=True, blank=True + ) def __str__(self) -> str: return self.name diff --git a/backend/events/serializers.py b/backend/events/serializers.py index 144c1f59a..50ac43e13 100644 --- a/backend/events/serializers.py +++ b/backend/events/serializers.py @@ -2,7 +2,7 @@ Serializers for the events app. """ -from typing import Dict, Union +from typing import Any, Dict, Union from django.utils.dateparse import parse_datetime from django.utils.translation import gettext as _ @@ -31,10 +31,43 @@ # MARK: Main Tables +class EventTextSerializer(serializers.ModelSerializer[EventText]): + class Meta: + model = EventText + fields = "__all__" + + class EventSerializer(serializers.ModelSerializer[Event]): + event_text = EventTextSerializer(read_only=True) + description = serializers.CharField(write_only=True, required=False) + class Meta: model = Event - fields = "__all__" + + extra_kwargs = { + "created_by": {"read_only": True}, + "social_links": {"required": False}, + "description": {"write_only": True}, + } + + fields = [ + "id", + "name", + "tagline", + "icon_url", + "type", + "online_location_link", + "offline_location", + "offline_location_lat", + "offline_location_long", + "created_by", + "social_links", + "is_private", + "start_time", + "end_time", + "event_text", + "description", + ] def validate(self, data: Dict[str, Union[str, int]]) -> Dict[str, Union[str, int]]: if parse_datetime(data["start_time"]) > parse_datetime(data["end_time"]): # type: ignore @@ -47,6 +80,16 @@ def validate(self, data: Dict[str, Union[str, int]]) -> Dict[str, Union[str, int return data + def create(self, validated_data: dict[str, Any]) -> Event: + description = validated_data.pop("description", None) + event = Event.objects.create(**validated_data) + + if event and description: + event_text = Event.objects.create(event_id=event, description=description) + event.event_text = event_text + + return event + class FormatSerializer(serializers.ModelSerializer[Event]): class Meta: @@ -115,12 +158,6 @@ class Meta: fields = "__all__" -class EventTextSerializer(serializers.ModelSerializer[EventText]): - class Meta: - model = EventText - fields = "__all__" - - class EventTopicSerializer(serializers.ModelSerializer[EventTopic]): class Meta: model = EventTopic diff --git a/backend/events/views.py b/backend/events/views.py index d1de501d4..fcd73c437 100644 --- a/backend/events/views.py +++ b/backend/events/views.py @@ -1,4 +1,8 @@ -from rest_framework import viewsets +from rest_framework import status, viewsets +from rest_framework.authentication import TokenAuthentication +from rest_framework.permissions import IsAuthenticatedOrReadOnly +from rest_framework.request import Request +from rest_framework.response import Response from rest_framework.throttling import ( AnonRateThrottle, UserRateThrottle, @@ -41,6 +45,76 @@ class EventViewSet(viewsets.ModelViewSet[Event]): serializer_class = EventSerializer pagination_class = CustomPagination throttle_classes = [AnonRateThrottle, UserRateThrottle] + permission_classes = [IsAuthenticatedOrReadOnly] + authentication_classes = [TokenAuthentication] + + def create(self, request: Request) -> Response: + serializer = self.get_serializer(data=request.data) + serializer.is_valid(raise_exception=True) + + return Response(serializer.data, status=status.HTTP_201_CREATED) + + def retrieve(self, request: Request, pk: str | None = None) -> Response: + if event := self.queryset.filter(id=pk).first(): + serializer = self.get_serializer(event) + + return Response(serializer.data, status=status.HTTP_200_OK) + + return Response({"error": "Event not found"}, status.HTTP_404_NOT_FOUND) + + def list(self, request: Request) -> Response: + serializer = self.get_serializer(self.get_queryset(), many=True) + + return Response(serializer.data, status=status.HTTP_200_OK) + + def update(self, request: Request, pk: str | None = None) -> Response: + event = self.queryset.filter(id=pk).first() + if event is None: + return Response({"error": "Event not found"}, status.HTTP_404_NOT_FOUND) + + if request.user != event.created_by: + return Response( + {"error": "You are not authorized to update this event"}, + status.HTTP_401_UNAUTHORIZED, + ) + + serializer = self.get_serializer(event, data=request.data, partial=True) + serializer.is_valid(raise_exception=True) + serializer.save() + + return Response(serializer.data, status.HTTP_200_OK) + + def partial_update(self, request: Request, pk: str | None = None) -> Response: + event = self.queryset.filter(id=pk).first() + if event is None: + return Response({"error": "Event not found"}, status.HTTP_404_NOT_FOUND) + + if request.user != event.created_by: + return Response( + {"error": "You are not authorized to update this event"}, + status.HTTP_401_UNAUTHORIZED, + ) + + serializer = self.get_serializer(event, data=request.data, partial=True) + serializer.is_valid(raise_exception=True) + serializer.save() + + return Response(serializer.data, status.HTTP_200_OK) + + def destroy(self, request: Request, pk: str | None = None) -> Response: + event = self.queryset.filter(id=pk).first() + if event is None: + return Response({"error": "Event not found"}, status.HTTP_404_NOT_FOUND) + + if request.user != event.created_by: + return Response( + {"error": "You are not authorized to delete this event"}, + status.HTTP_401_UNAUTHORIZED, + ) + + event.save() + + return Response({"message": "Event deleted successfully"}, status.HTTP_200_OK) class FormatViewSet(viewsets.ModelViewSet[Format]): diff --git a/frontend/assets/css/tailwind.css b/frontend/assets/css/tailwind.css index 76aa6984a..20319c392 100644 --- a/frontend/assets/css/tailwind.css +++ b/frontend/assets/css/tailwind.css @@ -131,6 +131,10 @@ @apply focus-brand elem-shadow-sm bg-light-layer-0 text-light-text hover:bg-light-highlight active:bg-light-layer-0 dark:border dark:border-dark-text dark:bg-dark-layer-0 dark:text-dark-text dark:hover:bg-dark-highlight dark:active:bg-dark-layer-0; } + .style-btns-next-to-one-another { + @apply mx-auto grid max-w-[70%] grid-cols-1 gap-y-4 sm:mx-0 sm:max-w-[90%] sm:grid-cols-2 sm:grid-rows-1 sm:gap-x-4 sm:gap-y-0 md:max-w-[70%] md:gap-x-6 lg:max-w-[60%] xl:max-w-[70%] xl:gap-x-8 2xl:max-w-[80%]; + } + .style-cta { @apply focus-brand border border-light-text bg-light-cta-orange fill-light-text text-light-text hover:bg-light-cta-orange/80 active:bg-light-cta-orange dark:border-dark-cta-orange dark:bg-dark-cta-orange/10 dark:fill-dark-cta-orange dark:text-dark-cta-orange dark:hover:bg-dark-cta-orange/25 dark:active:bg-dark-cta-orange/10; } diff --git a/frontend/components/landing/LandingContent.vue b/frontend/components/landing/LandingContent.vue index e3606a130..e7e5243e0 100644 --- a/frontend/components/landing/LandingContent.vue +++ b/frontend/components/landing/LandingContent.vue @@ -135,9 +135,7 @@ > {{ $t(text) }}

-
+
- - +
+ + +
-
+
diff --git a/frontend/pages/events/index.vue b/frontend/pages/events/index.vue index a70d5c430..810315598 100644 --- a/frontend/pages/events/index.vue +++ b/frontend/pages/events/index.vue @@ -13,23 +13,18 @@
- - +
diff --git a/frontend/stores/event.ts b/frontend/stores/event.ts index ce384537f..0ec3ff0b1 100644 --- a/frontend/stores/event.ts +++ b/frontend/stores/event.ts @@ -1,7 +1,10 @@ import type { Event, + EventText, PiniaResEvent, + PiniaResEvents, PiniaResEventText, + PiniaResEventTexts, } from "~/types/events/event"; interface EventStore { @@ -23,18 +26,23 @@ export const useEventStore = defineStore("event", { createdBy: "", iconURL: "", type: "learn", + onlineLocationLink: "", offlineLocation: "", + offlineLocationLat: "", + offlineLocationLong: "", getInvolvedURL: "", socialLinks: [""], startTime: "", + endTime: "", + creationDate: "", - // event_organizations organizations: [], - // event_text + eventTextID: "", description: "", getInvolved: "", }, + events: [], }), actions: { @@ -49,7 +57,7 @@ export const useEventStore = defineStore("event", { const [resEvent, resEventTexts] = await Promise.all([ useAsyncData( - async () => await fetchWithOptionalToken(`/entities/events/${id}`, {}) + async () => await fetchWithOptionalToken(`/events/events/${id}`, {}) ), // useAsyncData( // async () => @@ -68,7 +76,7 @@ export const useEventStore = defineStore("event", { useAsyncData( async () => await fetchWithOptionalToken( - `/entities/event_texts?event_id=${id}`, + `/events/event_texts?event_id=${id}`, {} ) ), @@ -101,7 +109,69 @@ export const useEventStore = defineStore("event", { // MARK: Fetch All - async fetchAll() {}, + async fetchAll() { + this.loading = true; + + const [responseEvents] = await Promise.all([ + useAsyncData( + async () => await fetchWithOptionalToken(`/events/events/`, {}) + ), + ]); + + const events = responseEvents.data as unknown as PiniaResEvents; + + console.log(`Here: ${JSON.stringify(events._value)}`); + + if (events._value) { + const responseEventTexts = (await Promise.all( + events._value.map((event) => + useAsyncData( + async () => + await fetchWithOptionalToken( + `/events/event_texts?event_id=${event.id}`, + {} + ) + ) + ) + )) as unknown as PiniaResEventTexts[]; + + const eventTextsData = responseEventTexts.map( + (text) => text.data._value.results[0] + ) as unknown as EventText[]; + + const eventsWithTexts = events._value.map( + (event: Event, index: number) => { + const texts = eventTextsData[index]; + return { + id: event.id, + name: event.name, + tagline: event.tagline, + createdBy: event.createdBy, + iconURL: event.iconURL, + type: event.type, + onlineLocationLink: event.onlineLocationLink, + offlineLocation: event.offlineLocation, + offlineLocationLat: event.offlineLocationLat, + offlineLocationLong: event.offlineLocationLong, + getInvolvedURL: event.getInvolvedURL, + socialLinks: event.socialLinks, + startTime: event.startTime, + endTime: event.endTime, + creationDate: event.creationDate, + organizations: event.organizations, + + eventTextID: event.eventTextID, + description: texts.description, + getInvolved: texts.getInvolved, + }; + } + ); + + this.events = eventsWithTexts; + } + + this.loading = false; + }, // MARK: Update @@ -109,6 +179,10 @@ export const useEventStore = defineStore("event", { // MARK: Delete - async delete() {}, + async delete(id: string | undefined) { + this.loading = true; + + this.loading = false; + }, }, }); diff --git a/frontend/stores/organization.ts b/frontend/stores/organization.ts index a7a23a67f..b0e9620ea 100644 --- a/frontend/stores/organization.ts +++ b/frontend/stores/organization.ts @@ -33,7 +33,7 @@ export const useOrganizationStore = defineStore("organization", { status: 1, groups: [], - organization_text_id: "", + organizationTextID: "", description: "", getInvolved: "", donationPrompt: "", @@ -145,7 +145,7 @@ export const useOrganizationStore = defineStore("organization", { this.organization.description = texts.description; this.organization.getInvolved = texts.getInvolved; this.organization.donationPrompt = texts.donationPrompt; - this.organization.organization_text_id = texts.id; + this.organization.organizationTextID = texts.id; this.loading = false; }, @@ -181,8 +181,6 @@ export const useOrganizationStore = defineStore("organization", { (text) => text.data._value.results[0] ) as unknown as OrganizationText[]; - console.log(`Here: ${JSON.stringify(orgTextsData)}`); - const organizationsWithTexts = orgs._value.map( (organization: Organization, index: number) => { const texts = orgTextsData[index]; @@ -197,7 +195,8 @@ export const useOrganizationStore = defineStore("organization", { socialLinks: organization.socialLinks, status: organization.status, groups: organization.groups, - organization_text_id: organization.organization_text_id, + + organizationTextID: organization.organizationTextID, description: texts.description, getInvolved: texts.getInvolved, donationPrompt: texts.donationPrompt, @@ -237,7 +236,7 @@ export const useOrganizationStore = defineStore("organization", { const responseOrgTexts = await $fetch( BASE_BACKEND_URL + - `/entities/organization_texts/${org.organization_text_id}/`, + `/entities/organization_texts/${org.organizationTextID}/`, { method: "PUT", body: { diff --git a/frontend/types/entities/organization.d.ts b/frontend/types/entities/organization.d.ts index 83f0d2d1d..42380eeb5 100644 --- a/frontend/types/entities/organization.d.ts +++ b/frontend/types/entities/organization.d.ts @@ -36,7 +36,7 @@ export interface Organization { // organization_task // task?: Task[]; - organization_text_id: string; + organizationTextID: string; description: string; getInvolved: string; donationPrompt: string; diff --git a/frontend/types/events/event.d.ts b/frontend/types/events/event.d.ts index 5cec4c85a..2bb67e826 100644 --- a/frontend/types/events/event.d.ts +++ b/frontend/types/events/event.d.ts @@ -30,7 +30,7 @@ export interface Event { // event_task // task?: Task[]; - // event_text + eventTextID: string; description: string; getInvolved: string; @@ -100,6 +100,13 @@ export interface PiniaResEvent { _value: Event; } +export interface PiniaResEvents { + __v_isShallow: boolean; + __v_isRef: boolean; + _rawValue: Event[]; + _value: Event[]; +} + export interface PiniaResEventText { __v_isShallow: boolean; __v_isRef: boolean; @@ -116,3 +123,39 @@ export interface PiniaResEventText { results: EventText[]; }; } + +export interface PiniaResEventTexts { + data: { + __v_isShallow: boolean; + __v_isRef: boolean; + _rawValue: { + count: number; + next: null; + previous: null; + results: EventText[]; + }; + _value: { + count: number; + next: null; + previous: null; + results: EventText[]; + }; + }; + pending: { + __v_isShallow: boolean; + __v_isRef: boolean; + _rawValue: boolean; + _value: boolean; + }; + error: { + _object: { [$key: string]: null }; + _key: string; + __v_isRef: boolean; + }; + status: { + __v_isShallow: boolean; + __v_isRef: boolean; + _rawValue: string; + _value: string; + }; +} diff --git a/frontend/utils/testEntities.ts b/frontend/utils/testEntities.ts index 3d805dfac..4b5bcd49a 100644 --- a/frontend/utils/testEntities.ts +++ b/frontend/utils/testEntities.ts @@ -48,7 +48,7 @@ export const testClimateOrg: Organization = { groups: ["Fundraising", "Campaigning"], socialLinks: ["climate-org@mastodon", "climate-org@email"], iconURL: "URL/for/image", - organization_text_id: "06cb36a3-13c5-4518-b676-33ec734744ed", + organizationTextID: "06cb36a3-13c5-4518-b676-33ec734744ed", description: "Testing how organizations work", getInvolved: "Hey, get involved!", donationPrompt: "Hey thanks!", @@ -119,7 +119,7 @@ export const testTechOrg: Organization = { groups: [testTechGroup1, testTechGroup2], socialLinks: ["tfb@mastodon", "tfb@email"], // donationPrompt: "Hey thanks!", - organization_text_id: "06cb36a3-13c5-4518-b676-33ec734744ed", + organizationTextID: "06cb36a3-13c5-4518-b676-33ec734744ed", description: "Testing how organizations work", getInvolved: "Hey, get involved!", donationPrompt: "Hey thanks!", @@ -145,6 +145,7 @@ export const testClimateEvent: Event = { // supportingUsers: [user, user], // iconURL: "/images/an_image.svg", socialLinks: ["climate_org@mastodon", "climate_org@email.com"], + eventTextID: "", }; export const testTechEvent: Event = { @@ -162,6 +163,7 @@ export const testTechEvent: Event = { startTime: new Date().toLocaleDateString(), // supportingUsers: [user, user, user], socialLinks: [""], + eventTextID: "", }; export const testResource: Resource = { From ad335c9642dfd31f1e74b1d776f2ccc5553a230a Mon Sep 17 00:00:00 2001 From: Andrew Tavis McAllister Date: Wed, 25 Sep 2024 01:19:07 +0200 Subject: [PATCH 6/8] Correct Event as EventText in create serializer --- backend/events/serializers.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/backend/events/serializers.py b/backend/events/serializers.py index 50ac43e13..5ecbd1024 100644 --- a/backend/events/serializers.py +++ b/backend/events/serializers.py @@ -85,7 +85,9 @@ def create(self, validated_data: dict[str, Any]) -> Event: event = Event.objects.create(**validated_data) if event and description: - event_text = Event.objects.create(event_id=event, description=description) + event_text = EventText.objects.create( + event_id=event, description=description + ) event.event_text = event_text return event From 2ab649e89377fe9c9d4eda6df1e8c2afce390d66 Mon Sep 17 00:00:00 2001 From: Andrew Tavis McAllister Date: Wed, 25 Sep 2024 01:27:23 +0200 Subject: [PATCH 7/8] Fixes to event views --- backend/entities/views.py | 1 + backend/events/views.py | 4 +++- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/backend/entities/views.py b/backend/entities/views.py index 5dec122df..38abc7f0d 100644 --- a/backend/entities/views.py +++ b/backend/entities/views.py @@ -112,6 +112,7 @@ def destroy(self, request: Request, *args: str, **kwargs: int) -> Response: ) group.delete() + return Response( {"message": "Group deleted successfully"}, status=status.HTTP_200_OK ) diff --git a/backend/events/views.py b/backend/events/views.py index fcd73c437..2ddf7378c 100644 --- a/backend/events/views.py +++ b/backend/events/views.py @@ -51,6 +51,8 @@ class EventViewSet(viewsets.ModelViewSet[Event]): def create(self, request: Request) -> Response: serializer = self.get_serializer(data=request.data) serializer.is_valid(raise_exception=True) + event = serializer.save(created_by=request.user) + Event.objects.create(id=event) return Response(serializer.data, status=status.HTTP_201_CREATED) @@ -112,7 +114,7 @@ def destroy(self, request: Request, pk: str | None = None) -> Response: status.HTTP_401_UNAUTHORIZED, ) - event.save() + event.delete() return Response({"message": "Event deleted successfully"}, status.HTTP_200_OK) From 37dc2596c34c36725b25f880a2f37c64f99e39cd Mon Sep 17 00:00:00 2001 From: Andrew Tavis McAllister Date: Wed, 25 Sep 2024 01:46:52 +0200 Subject: [PATCH 8/8] Update event views to incldue args and kwargs as required by mixins --- .../management/commands/populate_db.py | 6 ++-- backend/entities/views.py | 23 +++++++------ backend/events/views.py | 34 +++++++++---------- 3 files changed, 33 insertions(+), 30 deletions(-) diff --git a/backend/backend/management/commands/populate_db.py b/backend/backend/management/commands/populate_db.py index 00d373e0e..e799d7a94 100644 --- a/backend/backend/management/commands/populate_db.py +++ b/backend/backend/management/commands/populate_db.py @@ -60,7 +60,7 @@ def handle(self, *args: str, **options: Unpack[Options]) -> None: for o in range(num_orgs_per_user): user_org = OrganizationFactory( - name=f"{user_topic.name} Organization (U{u}-O{o})", + name=f"{user_topic.name} Organization (U{u}:O{o})", created_by=user, ) @@ -69,7 +69,7 @@ def handle(self, *args: str, **options: Unpack[Options]) -> None: for g in range(num_groups_per_org): user_org_group = GroupFactory( org_id=user_org, - name=f"{user_topic.name} Group (U{u}-O{o}-G{g})", + name=f"{user_topic.name} Group (U{u}:O{o}:G{g})", created_by=user, ) @@ -79,7 +79,7 @@ def handle(self, *args: str, **options: Unpack[Options]) -> None: for e in range(num_events_per_org): user_org_event = EventFactory( - name=f"{user_topic.name} Event (U{u}-O{o}-E{e})", + name=f"{user_topic.name} Event (U{u}:O{o}:E{e})", type=random.choice(["learn", "action"]), created_by=user, ) diff --git a/backend/entities/views.py b/backend/entities/views.py index 38abc7f0d..a4b7f1852 100644 --- a/backend/entities/views.py +++ b/backend/entities/views.py @@ -67,15 +67,17 @@ def create(self, request: Request) -> Response: serializer = self.get_serializer(data=request.data) serializer.is_valid(raise_exception=True) serializer.save(created_by=request.user) - data = {"message": f"New Group created: {serializer.data}"} + data = {"message": f"New group created: {serializer.data}"} return Response(data, status=status.HTTP_201_CREATED) def retrieve(self, request: Request, *args: str, **kwargs: int) -> Response: - group = self.queryset.get(id=kwargs["pk"]) - serializer = self.get_serializer(group) + if group := self.queryset.get(id=kwargs["pk"]): + serializer = self.get_serializer(group) - return Response(serializer.data, status=status.HTTP_200_OK) + return Response(serializer.data, status=status.HTTP_200_OK) + + return Response({"error": "Group not found"}, status.HTTP_404_NOT_FOUND) def partial_update(self, request: Request, *args: str, **kwargs: int) -> Response: group = self.queryset.filter(id=kwargs["pk"]).first() @@ -126,13 +128,19 @@ class OrganizationViewSet(viewsets.ModelViewSet[Organization]): permission_classes = [IsAuthenticatedOrReadOnly] authentication_classes = [TokenAuthentication] + def list(self, request: Request) -> Response: + serializer = self.get_serializer(self.get_queryset(), many=True) + + return Response(serializer.data, status=status.HTTP_200_OK) + def create(self, request: Request) -> Response: serializer = self.get_serializer(data=request.data) serializer.is_valid(raise_exception=True) org = serializer.save(created_by=request.user) OrganizationApplication.objects.create(org_id=org) + data = {"message": f"New organization created: {serializer.data}"} - return Response(serializer.data, status=status.HTTP_201_CREATED) + return Response(data, status=status.HTTP_201_CREATED) def retrieve(self, request: Request, pk: str | None = None) -> Response: if org := self.queryset.filter(id=pk).first(): @@ -142,11 +150,6 @@ def retrieve(self, request: Request, pk: str | None = None) -> Response: return Response({"error": "Organization not found"}, status.HTTP_404_NOT_FOUND) - def list(self, request: Request) -> Response: - serializer = self.get_serializer(self.get_queryset(), many=True) - - return Response(serializer.data, status=status.HTTP_200_OK) - def update(self, request: Request, pk: str | None = None) -> Response: org = self.queryset.filter(id=pk).first() if org is None: diff --git a/backend/events/views.py b/backend/events/views.py index 2ddf7378c..1713a1c2b 100644 --- a/backend/events/views.py +++ b/backend/events/views.py @@ -48,29 +48,29 @@ class EventViewSet(viewsets.ModelViewSet[Event]): permission_classes = [IsAuthenticatedOrReadOnly] authentication_classes = [TokenAuthentication] - def create(self, request: Request) -> Response: + def list(self, request: Request, *args: str, **kwargs: int) -> Response: + serializer = self.get_serializer(self.get_queryset(), many=True) + + return Response(serializer.data, status=status.HTTP_200_OK) + + def create(self, request: Request, *args: str, **kwargs: int) -> Response: serializer = self.get_serializer(data=request.data) serializer.is_valid(raise_exception=True) - event = serializer.save(created_by=request.user) - Event.objects.create(id=event) + serializer.save(created_by=request.user) + data = {"message": f"New event created: {serializer.data}"} - return Response(serializer.data, status=status.HTTP_201_CREATED) + return Response(data, status=status.HTTP_201_CREATED) - def retrieve(self, request: Request, pk: str | None = None) -> Response: - if event := self.queryset.filter(id=pk).first(): + def retrieve(self, request: Request, *args: str, **kwargs: int) -> Response: + if event := self.queryset.get(id=kwargs["pk"]): serializer = self.get_serializer(event) return Response(serializer.data, status=status.HTTP_200_OK) return Response({"error": "Event not found"}, status.HTTP_404_NOT_FOUND) - def list(self, request: Request) -> Response: - serializer = self.get_serializer(self.get_queryset(), many=True) - - return Response(serializer.data, status=status.HTTP_200_OK) - - def update(self, request: Request, pk: str | None = None) -> Response: - event = self.queryset.filter(id=pk).first() + def update(self, request: Request, *args: str, **kwargs: int) -> Response: + event = self.queryset.filter(id=kwargs["pk"]).first() if event is None: return Response({"error": "Event not found"}, status.HTTP_404_NOT_FOUND) @@ -86,8 +86,8 @@ def update(self, request: Request, pk: str | None = None) -> Response: return Response(serializer.data, status.HTTP_200_OK) - def partial_update(self, request: Request, pk: str | None = None) -> Response: - event = self.queryset.filter(id=pk).first() + def partial_update(self, request: Request, *args: str, **kwargs: int) -> Response: + event = self.queryset.filter(id=kwargs["pk"]).first() if event is None: return Response({"error": "Event not found"}, status.HTTP_404_NOT_FOUND) @@ -103,8 +103,8 @@ def partial_update(self, request: Request, pk: str | None = None) -> Response: return Response(serializer.data, status.HTTP_200_OK) - def destroy(self, request: Request, pk: str | None = None) -> Response: - event = self.queryset.filter(id=pk).first() + def destroy(self, request: Request, *args: str, **kwargs: int) -> Response: + event = self.queryset.filter(id=kwargs["pk"]).first() if event is None: return Response({"error": "Event not found"}, status.HTTP_404_NOT_FOUND)