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