diff --git a/frontend/public/locales/en/common.json b/frontend/public/locales/en/common.json index ad3e60bf4a48f4680879a02d95555feb3028f70c..795e38948ada4b7c180db25a267d72d19dc00310 100644 --- a/frontend/public/locales/en/common.json +++ b/frontend/public/locales/en/common.json @@ -48,7 +48,6 @@ "registerText": "Please search for e-mail or phone number before registering a new guest to prevent duplicates.", "registerButtonText": "Register new guest" }, - "loading": "Loading...", "termsHeader": "Terms", "staging": "Staging", @@ -94,7 +93,8 @@ "next": "Next", "save": "Save", "cancel": "Cancel", - "backToFrontPage": "Go to front page" + "backToFrontPage": "Go to front page", + "cancelInvitation": "Cancel" }, "registerWizardText": { "registerPage": "Enter the contact information for the guest below. All fields are mandatory.", diff --git a/frontend/public/locales/nb/common.json b/frontend/public/locales/nb/common.json index 41828b97a9f79695f3254220e07477295699658d..2a1a32b4b867ecc9356deac7db1376812f4fed99 100644 --- a/frontend/public/locales/nb/common.json +++ b/frontend/public/locales/nb/common.json @@ -93,7 +93,8 @@ "next": "Neste", "save": "Lagre", "cancel": "Avbryt", - "backToFrontPage": "Tilbake til forsiden" + "backToFrontPage": "Tilbake til forsiden", + "cancelInvitation": "Kanseller" }, "registerWizardText": { "registerPage": "Fyll inn kontaktinformasjonen til gjesten under. Alle feltene er obligatoriske.", diff --git a/frontend/public/locales/nn/common.json b/frontend/public/locales/nn/common.json index af1b8aac7174d127b2ae632e014d7c24eec139f4..30dde00a616e4214513de8a3ce1d62e3049e09dd 100644 --- a/frontend/public/locales/nn/common.json +++ b/frontend/public/locales/nn/common.json @@ -94,7 +94,8 @@ "next": "Neste", "save": "Lagre", "cancel": "Avbryt", - "backToFrontPage": "Tilbake til forsida" + "backToFrontPage": "Tilbake til forsida", + "cancelInvitation": "Kanseller" }, "registerWizardText": { "registerPage": "Fyll inn kontaktinformasjonen til gjesten under. Alle feltene er obligatoriske.", diff --git a/frontend/src/routes/sponsor/frontpage/index.tsx b/frontend/src/routes/sponsor/frontpage/index.tsx index d20af82b1dd33929132485347856ff5d4062d075..b682da5bbe4a8c1745fc2dac6e1ca809160d207a 100644 --- a/frontend/src/routes/sponsor/frontpage/index.tsx +++ b/frontend/src/routes/sponsor/frontpage/index.tsx @@ -1,4 +1,4 @@ -import { useState } from 'react' +import React, { useState } from 'react' import { Table, TableBody, @@ -23,15 +23,25 @@ import SponsorGuestButtons from '../../components/sponsorGuestButtons' interface GuestProps { persons: Guest[] + // eslint-disable-next-line react/no-unused-prop-types + cancelCallback?: (roleId: string) => void } interface PersonLineProps { person: Guest role: Role showStatusColumn?: boolean + displayCancel?: boolean + cancelCallback?: (roleId: string) => void } -const PersonLine = ({ person, role, showStatusColumn }: PersonLineProps) => { +const PersonLine = ({ + person, + role, + showStatusColumn, + displayCancel, + cancelCallback, +}: PersonLineProps) => { const [t, i18n] = useTranslation(['common']) return ( @@ -65,13 +75,27 @@ const PersonLine = ({ person, role, showStatusColumn }: PersonLineProps) => { {i18n.language === 'en' ? role.ou_en : role.ou_nb} </TableCell> <TableCell align="left"> - <Button - variant="contained" - component={Link} - to={`/sponsor/guest/${person.pid}`} - > - {t('common:details')} - </Button> + {displayCancel ? ( + <Button + data-testid="button-invite-cancel" + sx={{ color: 'theme.palette.secondary', mr: 1 }} + onClick={() => { + if (cancelCallback) { + cancelCallback(role.id) + } + }} + > + {t('common:button.cancel')} + </Button> + ) : ( + <Button + variant="contained" + component={Link} + to={`/sponsor/guest/${person.pid}`} + > + {t('common:details')} + </Button> + )} </TableCell> </TableRow> ) @@ -79,9 +103,15 @@ const PersonLine = ({ person, role, showStatusColumn }: PersonLineProps) => { PersonLine.defaultProps = { showStatusColumn: false, + displayCancel: false, + // eslint-disable-next-line @typescript-eslint/no-unused-vars + cancelCallback: (roleId: number) => {}, } -const WaitingForGuestRegistration = ({ persons }: GuestProps) => { +const WaitingForGuestRegistration = ({ + persons, + cancelCallback, +}: GuestProps) => { const [activeExpanded, setActiveExpanded] = useState(false) // Show guests that have not responded to the invite yet @@ -119,7 +149,12 @@ const WaitingForGuestRegistration = ({ persons }: GuestProps) => { guests.map((person) => person.roles ? ( person.roles.map((role) => ( - <PersonLine role={role} person={person} /> + <PersonLine + role={role} + person={person} + displayCancel + cancelCallback={cancelCallback} + /> )) ) : ( <></> @@ -142,6 +177,11 @@ const WaitingForGuestRegistration = ({ persons }: GuestProps) => { ) } +WaitingForGuestRegistration.defaultProps = { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + cancelCallback: (roleId: number) => {}, +} + const ActiveGuests = ({ persons }: GuestProps) => { const [activeExpanded, setActiveExpanded] = useState(false) @@ -209,6 +249,11 @@ const ActiveGuests = ({ persons }: GuestProps) => { ) } +ActiveGuests.defaultProps = { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + cancelCallback: (roleId: number) => {}, +} + const WaitingGuests = ({ persons }: GuestProps) => { const [waitingExpanded, setWaitingExpanded] = useState(false) @@ -271,15 +316,24 @@ const WaitingGuests = ({ persons }: GuestProps) => { ) } +WaitingGuests.defaultProps = { + // eslint-disable-next-line @typescript-eslint/no-unused-vars + cancelCallback: (roleId: number) => {}, +} + interface FrontPageProps { guests: Guest[] + cancelRole: (roleId: string) => void } -function FrontPage({ guests }: FrontPageProps) { +function FrontPage({ guests, cancelRole }: FrontPageProps) { return ( <Page> <SponsorGuestButtons yourGuestsActive /> - <WaitingForGuestRegistration persons={guests} /> + <WaitingForGuestRegistration + persons={guests} + cancelCallback={cancelRole} + /> <WaitingGuests persons={guests} /> <ActiveGuests persons={guests} /> </Page> diff --git a/frontend/src/routes/sponsor/index.tsx b/frontend/src/routes/sponsor/index.tsx index d070e2b28d5d4cef3fb69234c66d4c6335ebd40f..741bda77f5085ce0aee13de1a09a6dc39d2fbf1f 100644 --- a/frontend/src/routes/sponsor/index.tsx +++ b/frontend/src/routes/sponsor/index.tsx @@ -3,7 +3,7 @@ import { Route } from 'react-router-dom' import FrontPage from 'routes/sponsor/frontpage' import { FetchedGuest, Guest } from 'interfaces' -import { parseRole } from 'utils' +import { parseRole, submitJsonOpts } from 'utils' import GuestRoutes from './guest' function Sponsor() { @@ -37,6 +37,29 @@ function Sponsor() { } } + const cancelRole = (roleId: string) => { + // There is no body for this request, but using submitJsonOpts still to + // set the CSRF-token + fetch(`/api/ui/v1/invite/?role_id=${roleId}`, submitJsonOpts('DELETE', {})) + .then((res) => { + if (!res.ok) { + return null + } + return res.text() + }) + .then((result) => { + if (result !== null) { + // The invitation has been removed. Just reload all the data form the server to get updated data. + // The guests state will be updated by getGuestsInfo and this will trigger a rerender + getGuestsInfo() + } + }) + .catch((error) => { + // TODO User should get some feedback telling him something failed + console.log('error', error) + }) + } + useEffect(() => { getGuestsInfo() }, []) @@ -47,7 +70,7 @@ function Sponsor() { <GuestRoutes /> </Route> <Route exact path="/sponsor"> - <FrontPage guests={guests} /> + <FrontPage guests={guests} cancelRole={cancelRole} /> </Route> </> ) diff --git a/gregui/api/urls.py b/gregui/api/urls.py index af6ff1ffdd2ee35a2284878d3147d0fc77fe1b2a..46e1cf0fbe918a4bf033cc9b52d133fb0bea1b79 100644 --- a/gregui/api/urls.py +++ b/gregui/api/urls.py @@ -4,7 +4,7 @@ from rest_framework.routers import DefaultRouter from gregui.api.views.invitation import ( CheckInvitationView, - CreateInvitationView, + InvitationView, InvitedGuestView, ) from gregui.api.views.person import PersonSearchView, PersonView @@ -21,7 +21,7 @@ urlpatterns += [ re_path(r"units/$", UnitsViewSet.as_view(), name="units"), path("invited/", InvitedGuestView.as_view(), name="invited-info"), path("invited/<uuid>", CheckInvitationView.as_view(), name="invite-verify"), - path("invite/", CreateInvitationView.as_view(), name="invite-create"), + path("invite/", InvitationView.as_view(), name="invitation"), path("person/<int:id>", PersonView.as_view(), name="person-get"), path( "person/search/<searchstring>", PersonSearchView.as_view(), name="person-search" diff --git a/gregui/api/views/invitation.py b/gregui/api/views/invitation.py index 5ffca10acd3f5643e16a82654153a29fa12fea66..2fd4485434a29cc01ceda0b9ee8337dd2f264b1c 100644 --- a/gregui/api/views/invitation.py +++ b/gregui/api/views/invitation.py @@ -7,7 +7,7 @@ from django.http.response import JsonResponse from django.utils import timezone from rest_framework import status from rest_framework.authentication import SessionAuthentication, BasicAuthentication -from rest_framework.generics import CreateAPIView, GenericAPIView +from rest_framework.generics import CreateAPIView, GenericAPIView, DestroyAPIView from rest_framework.parsers import JSONParser from rest_framework.permissions import AllowAny from rest_framework.response import Response @@ -20,9 +20,9 @@ from gregui.api.serializers.invitation import InviteGuestSerializer from gregui.models import GregUserProfile -class CreateInvitationView(CreateAPIView): +class InvitationView(CreateAPIView, DestroyAPIView): """ - Invitation creation endpoint + Endpoint for invitation creation and cancelling { @@ -72,6 +72,26 @@ class CreateInvitationView(CreateAPIView): print(invitationlink) return Response(status=status.HTTP_201_CREATED) + def delete(self, request, *args, **kwargs) -> Response: + role_id = request.query_params["role_id"] + invitationlink = InvitationLink.objects.get(invitation__role_id=int(role_id)) + + # TODO Determine if person should be deleted as well + if invitationlink: + if ( + invitationlink.invitation.role.person.is_registered + or invitationlink.invitation.role.person.is_verified + ): + # The guest has already gone through the registration step. The guest should + # not be verified, but including that check just in case here + return Response(status.HTTP_400_BAD_REQUEST) + + # Delete the role, the cascading will cause all the invitation links connected + # to it to be removed as well + invitationlink.invitation.role.delete() + + return Response(status=status.HTTP_200_OK) + class CheckInvitationView(APIView): authentication_classes = [] diff --git a/gregui/tests/api/test_invite_guest.py b/gregui/tests/api/test_invite_guest.py index 87554ec7edbb03b6325945abc06e478e469ba6dc..0f265b1e44311d0001c0d6e7f444f3c6305a1d42 100644 --- a/gregui/tests/api/test_invite_guest.py +++ b/gregui/tests/api/test_invite_guest.py @@ -4,8 +4,8 @@ from rest_framework import status from rest_framework.reverse import reverse from rest_framework.test import APIRequestFactory, force_authenticate -from greg.models import Identity, Person -from gregui.api.views.invitation import CreateInvitationView +from greg.models import Identity, Person, Role, Invitation, InvitationLink +from gregui.api.views.invitation import InvitationView @pytest.mark.django_db @@ -25,7 +25,7 @@ def test_invite_guest(client, user_sponsor, unit_foo, role_type_foo): "type": role_type_foo.id, }, } - url = reverse("gregui-v1:invite-create") + url = reverse("gregui-v1:invitation") all_persons = Person.objects.all() assert len(all_persons) == 0 @@ -34,7 +34,7 @@ def test_invite_guest(client, user_sponsor, unit_foo, role_type_foo): request = factory.post(url, data, format="json") force_authenticate(request, user=user_sponsor) - view = CreateInvitationView.as_view() + view = InvitationView.as_view() response = view(request) assert response.status_code == status.HTTP_201_CREATED @@ -49,3 +49,23 @@ def test_invite_guest(client, user_sponsor, unit_foo, role_type_foo): type=Identity.IdentityType.PRIVATE_EMAIL, value="test@example.com", ).exists() + + +@pytest.mark.django_db +def test_invite_cancel( + client, invitation_link, invitation, role, log_in, sponsor_guy, user_sponsor +): + # TODO: Should all sponsors be allowed to delete arbitrary invitations? + log_in(user_sponsor) + url = reverse("gregui-v1:invitation") + + # Check that the role is there + role = Role.objects.get(id=role.id) + response = client.delete("%s?role_id=%s" % (url, str(role.id))) + + assert response.status_code == status.HTTP_200_OK + + # The role, invitation and connected links should now have been removed + assert Role.objects.filter(id=role.id).count() == 0 + assert Invitation.objects.filter(id=invitation.id).count() == 0 + assert InvitationLink.objects.filter(invitation__id=invitation.id).count() == 0 diff --git a/gregui/tests/conftest.py b/gregui/tests/conftest.py index 1bd22154f3a68dc909a2eebb94a8a65aee17f9bc..fc70bdca4858316b191b6c1d1e7b7860363521a3 100644 --- a/gregui/tests/conftest.py +++ b/gregui/tests/conftest.py @@ -9,7 +9,6 @@ from django.utils.timezone import make_aware from rest_framework.authtoken.admin import User from rest_framework.test import APIClient - from greg.models import ( Invitation, InvitationLink, @@ -26,6 +25,7 @@ from gregui.models import GregUserProfile # see https://github.com/joke2k/faker/issues/753 logging.getLogger("faker").setLevel(logging.ERROR) + # OIDC stuff @pytest.fixture def claims(): @@ -246,9 +246,8 @@ def greg_sponsors(data): @pytest.fixture -def log_in(client, greg_users): - def _log_in(username): - user = greg_users[username] +def log_in(client): + def _log_in(user): client.force_login(user=user) # It seems like the session was not updated automatically this way session = client.session