Ver Fonte

Switch to a migration-friendly CoderedStreamField for internal concrete models (#496)

Vince Salvino há 2 anos atrás
pai
commit
a44e5e49d7

+ 63 - 1
coderedcms/fields.py

@@ -1,10 +1,69 @@
 from django.db import models
+from django.forms.widgets import Textarea
+from wagtail.core.fields import StreamField
 
 from coderedcms.widgets import ColorPickerWidget
-from django.forms.widgets import Textarea
+
+
+class CoderedStreamField(StreamField):
+    """
+    An exact copy of the Wagtail StreamField, modified to NOT PRESERVE HISTORY
+    in Django migrations.
+
+    Since our StreamFields are generally huge, and we also let sites override
+    the blocks in our concrete models dynamically, this creates a slew of
+    migration problems (most commonly: a client overrides CODERED_FRONTEND_*,
+    which changes a string used in a concrete model, which triggers a migration
+    back in coderedcms). Eliminiating the blocks from the deconstructed
+    StreamField allows us to have dynamic streamfields without breaking
+    migrations or having to refactor the core concepts of this package.
+
+    Internally, we should ALWAYS use CoderedStreamField on CONCRETE models,
+    meaning models which are part of our package and saved to the database. For
+    ABSTRACT models - meaning they are made concrete in the client site - we
+    should continue to use Wagtail StreamField to keep things Wagtail-ish by
+    default. The client may then decide to use CoderedStreamField if they are
+    annoyed by the big migrations.
+
+    CAVEAT EMPTOR:
+
+    Client sites built with CRX may use this in place of the Wagtail
+    StreamField, in order to avoid huge migration files. However, note that it
+    will not be possible to mine data out of a CoderedStreamField during a
+    migration (e.g. RunPython).
+
+    Inspired by:
+    https://cynthiakiser.com/blog/2022/01/06/trimming-wagtail-migration-cruft.html
+    """
+    def __init__(self, *args, **kwargs):
+        """
+        Patch init to work around django reconstruct not sending empty args.
+        """
+        # If we did not get an arg, pass an empty list through to the parent.
+        if not args:
+            args = [[]]
+        return super().__init__(*args, **kwargs)
+
+    def deconstruct(self):
+        """
+        Override to ignore any blocks within the StreamField when
+        decustructing into a migration.
+
+        The output should look something like this, regardless of how many
+        blocks are nested within the StreamField::
+
+            ("body", coderedcms.fields.CoderedStreamField([]))
+
+        """
+        name, path, block_types, kwargs = super().deconstruct()
+        block_types = []
+        return name, path, block_types, kwargs
 
 
 class ColorField(models.CharField):
+    """
+    A CharField which uses the HTML5 color picker widget.
+    """
     def __init__(self, *args, **kwargs):
         kwargs['max_length'] = 255
         super().__init__(*args, **kwargs)
@@ -15,6 +74,9 @@ class ColorField(models.CharField):
 
 
 class MonospaceField(models.TextField):
+    """
+    A TextField which renders as a large HTML textarea with monospace font.
+    """
     def formfield(self, **kwargs):
         kwargs["widget"] = Textarea(attrs={
             "rows": 12,

Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0001_initial.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0002_auto_20180829_1538.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0003_auto_20180912_1632.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0005_auto_20181214_2214.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0008_auto_20190122_1242.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0010_auto_20190320_1231.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0013_pagepreview_templates.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0014_classifiers.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0016_auto_20191025_1620.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0018_auto_20200805_1702.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0019_spelling_corrections.py


Diff do ficheiro suprimidas por serem muito extensas
+ 1 - 0
coderedcms/migrations/0026_auto_20220513_1627.py


+ 2 - 2
coderedcms/models/page_models.py

@@ -69,7 +69,7 @@ from coderedcms.blocks import (
     STREAMFORM_BLOCKS,
     ContentWallBlock,
 )
-from coderedcms.fields import ColorField
+from coderedcms.fields import CoderedStreamField, ColorField
 from coderedcms.forms import CoderedFormBuilder, CoderedSubmissionsListView
 from coderedcms.models.snippet_models import ClassifierTerm
 from coderedcms.models.wagtailsettings_models import (
@@ -259,7 +259,7 @@ class CoderedPage(WagtailCacheMixin, SeoMixin, Page, metaclass=CoderedPageMeta):
     # Settings
     ###############
 
-    content_walls = StreamField(
+    content_walls = CoderedStreamField(
         [
             ('content_wall', ContentWallBlock())
         ],

+ 11 - 7
coderedcms/models/snippet_models.py

@@ -12,13 +12,13 @@ from wagtail.admin.edit_handlers import (
     InlinePanel,
     MultiFieldPanel,
     StreamFieldPanel)
-from wagtail.core.fields import StreamField
 from wagtail.core.models import Orderable
 from wagtail.images.edit_handlers import ImageChooserPanel
 from wagtail.snippets.models import register_snippet
 from wagtail.images import get_image_model_string
 
 from coderedcms.blocks import HTML_STREAMBLOCKS, LAYOUT_STREAMBLOCKS, NAVIGATION_STREAMBLOCKS
+from coderedcms.fields import CoderedStreamField
 from coderedcms.settings import cr_settings
 
 
@@ -112,7 +112,7 @@ class CarouselSlide(Orderable, models.Model):
         verbose_name=_('Custom ID'),
     )
 
-    content = StreamField(HTML_STREAMBLOCKS, blank=True)
+    content = CoderedStreamField(HTML_STREAMBLOCKS, blank=True)
 
     panels = (
         [
@@ -235,9 +235,10 @@ class Navbar(models.Model):
         blank=True,
         verbose_name=_('Custom ID'),
     )
-    menu_items = StreamField(
+    menu_items = CoderedStreamField(
         NAVIGATION_STREAMBLOCKS,
         verbose_name=_('Navigation links'),
+        blank=True,
     )
 
     panels = [
@@ -278,9 +279,10 @@ class Footer(models.Model):
         blank=True,
         verbose_name=_('Custom ID'),
     )
-    content = StreamField(
+    content = CoderedStreamField(
         LAYOUT_STREAMBLOCKS,
         verbose_name=_('Content'),
+        blank=True,
     )
 
     panels = [
@@ -312,9 +314,10 @@ class ReusableContent(models.Model):
         max_length=255,
         verbose_name=_('Name'),
     )
-    content = StreamField(
+    content = CoderedStreamField(
         LAYOUT_STREAMBLOCKS,
-        verbose_name=_('content')
+        verbose_name=_('content'),
+        blank=True,
     )
 
     panels = [
@@ -338,9 +341,10 @@ class ContentWall(models.Model):
         max_length=255,
         verbose_name=_('Name'),
     )
-    content = StreamField(
+    content = CoderedStreamField(
         LAYOUT_STREAMBLOCKS,
         verbose_name=_('Content'),
+        blank=True,
     )
     is_dismissible = models.BooleanField(
         default=True,

Alguns ficheiros não foram mostrados porque muitos ficheiros mudaram neste diff