Skip to content

Commit

Permalink
Make get_client_root_url overridable both in both mixins
Browse files Browse the repository at this point in the history
  • Loading branch information
zerolab committed Feb 15, 2024
1 parent e2422e7 commit 5bfa8db
Showing 1 changed file with 54 additions and 26 deletions.
80 changes: 54 additions & 26 deletions src/wagtail_headless_preview/models.py
Original file line number Diff line number Diff line change
@@ -1,16 +1,24 @@
import datetime
import json

from typing import TYPE_CHECKING, Optional

from django.contrib.contenttypes.models import ContentType
from django.core.signing import TimestampSigner
from django.db import models
from django.shortcuts import redirect, render
from django.utils.http import urlencode
from wagtail.models import Site

from wagtail_headless_preview.settings import headless_preview_settings
from wagtail_headless_preview.signals import preview_update


if TYPE_CHECKING:
from django.http import HttpRequest
from wagtail.models import Page, Site


class PagePreview(models.Model):
token = models.CharField(max_length=255, unique=True)
content_type = models.ForeignKey(
Expand All @@ -19,7 +27,7 @@ class PagePreview(models.Model):
content_json = models.TextField()
created_at = models.DateField(auto_now_add=True)

def as_page(self):
def as_page(self) -> "Page":
content = json.loads(self.content_json)
page_model = ContentType.objects.get_for_id(
content["content_type"]
Expand All @@ -34,7 +42,7 @@ def garbage_collect(cls):
cls.objects.filter(created_at__lt=yesterday).delete()


def get_client_root_url_from_site(site):
def get_client_root_url_from_site(site) -> str:
try:
root_url = headless_preview_settings.CLIENT_URLS[site.hostname]
except (AttributeError, KeyError):
Expand All @@ -48,17 +56,38 @@ def get_client_root_url_from_site(site):
return root_url


class HeadlessPreviewMixin:
class HeadlessBase:
def _get_site_from_request(self, request: "HttpRequest") -> "Site":
"""
Copy of Page.get_site() which passes the request object to Page.get_url_parts()
"""
url_parts = self.get_url_parts(request=request)
if url_parts is None:
# page is not routable
return

site_id, root_url, page_path = url_parts

return Site.objects.get(id=site_id)

def get_client_root_url(self, request: "HttpRequest") -> str:
return get_client_root_url_from_site(self._get_site_from_request(request))

class Meta:
abstract = True


class HeadlessPreviewMixin(HeadlessBase):
@classmethod
def get_preview_signer(cls):
def get_preview_signer(cls) -> TimestampSigner:
return TimestampSigner(salt="headlesspreview.token")

@classmethod
def get_content_type_str(cls):
def get_content_type_str(cls) -> str:
return cls._meta.app_label + "." + cls.__name__.lower()

@classmethod
def get_page_from_preview_token(cls, token):
def get_page_from_preview_token(cls, token: str) -> Optional["Page"]:
content_type = ContentType.objects.get_for_model(cls)

# Check token is valid
Expand All @@ -71,7 +100,16 @@ def get_page_from_preview_token(cls, token):
except PagePreview.DoesNotExist:
return

def create_page_preview(self):
def update_page_preview(self, token: str) -> PagePreview:
return PagePreview.objects.update_or_create(
token=token,
defaults={
"content_type": self.content_type,
"content_json": self.to_json(),
},
)

def create_page_preview(self) -> PagePreview:
if self.pk is None:
identifier = (
f"parent_id={self.get_parent().pk};page_type={self._meta.label}"
Expand All @@ -86,34 +124,22 @@ def create_page_preview(self):

return preview

def update_page_preview(self, token):
return PagePreview.objects.update_or_create(
token=token,
defaults={
"content_type": self.content_type,
"content_json": self.to_json(),
},
)

def get_client_root_url(self):
return get_client_root_url_from_site(self.get_site())

def get_preview_url(self, token):
def get_preview_url(self, request: "HttpRequest", token: str) -> str:
return (
self.get_client_root_url()
self.get_client_root_url(request)
+ "?"
+ urlencode({"content_type": self.get_content_type_str(), "token": token})
)

def serve_preview(self, request, preview_mode):
def serve_preview(self, request: "HttpRequest", preview_mode):
PagePreview.garbage_collect()
page_preview = self.create_page_preview()
page_preview.save()

# Send the preview_update signal. Other apps can implement their own handling
preview_update.send(sender=HeadlessPreviewMixin, token=page_preview.token)

preview_url = self.get_preview_url(page_preview.token)
preview_url = self.get_preview_url(request, page_preview.token)
if headless_preview_settings.REDIRECT_ON_PREVIEW:
return redirect(preview_url)

Expand All @@ -126,8 +152,8 @@ def serve_preview(self, request, preview_mode):
return response


class HeadlessServeMixin:
def serve(self, request):
class HeadlessServeMixin(HeadlessBase):
def serve(self, request: "HttpRequest"):
"""
Mixin overriding the default serve method with a redirect.
The URL of the requested page is kept the same, only the host is
Expand All @@ -136,10 +162,12 @@ def serve(self, request):
However, you can enforce a single host using the HEADLESS_SERVE_BASE_URL
setting.
"""

if headless_preview_settings.SERVE_BASE_URL:
base_url = headless_preview_settings.SERVE_BASE_URL
else:
base_url = get_client_root_url_from_site(self.get_site())
base_url = self.get_client_root_url(request)

site_id, site_root, relative_page_url = self.get_url_parts(request)
return redirect(f"{base_url.rstrip('/')}{relative_page_url}")

Expand Down

0 comments on commit 5bfa8db

Please sign in to comment.