Matthias Kestenholz: Posts about Djangohttps://406.ch/writing/category-django/2025-01-17T12:00:00ZMatthias KestenholzDjango admin tip: Adding links to related objects in change formshttps://406.ch/writing/django-admin-tip-adding-links-to-related-objects-in-change-forms/2025-01-17T12:00:00Z2025-01-17T12:00:00Z<h1 id="django-admin-tip-adding-links-to-related-objects-in-change-forms"><a class="toclink" href="#django-admin-tip-adding-links-to-related-objects-in-change-forms">Django admin tip: Adding links to related objects in change forms</a></h1> <p>Any issue which came up on the Django Forum and Discord is how to add links to other objects to the Django administration interface. It&rsquo;s something I&rsquo;m doing often and I want to share the pattern I&rsquo;m using.</p> <p>It&rsquo;s definitely not rocket science and there are probably better ways to do it, but this one works well for me.</p> <h2 id="method-1-override-the-change-form-template"><a class="toclink" href="#method-1-override-the-change-form-template">Method 1: Override the change form template</a></h2> <p>In one project users can be the editor of exactly one organization. The link between organizations and users is achieved using a <code>Editor</code> model with a <code>ForeignKey(Organization)</code> and a <code>OneToOneField(User)</code>.</p> <p>I wanted to add a link to the organization page at the bottom of the user form. An easy way to achieve this is to add a template at <code>templates/admin/auth/user/change_form.html</code> (or something similar if you&rsquo;re using a custom user model):</p> <div class="chl"><pre><span></span><code><span class="cp">{%</span> <span class="k">extends</span> <span class="s2">&quot;admin/change_form.html&quot;</span> <span class="cp">%}</span> <span class="cp">{%</span> <span class="k">block</span> <span class="nv">after_related_objects</span> <span class="cp">%}</span> <span class="cp">{{</span> <span class="nb">block</span><span class="nv">.super</span> <span class="cp">}}</span> <span class="cp">{%</span> <span class="k">if</span> <span class="nv">original.editor</span> <span class="cp">%}</span> <span class="p">&lt;</span><span class="nt">fieldset</span> <span class="na">class</span><span class="o">=</span><span class="s">&quot;module aligned&quot;</span><span class="p">&gt;</span> <span class="p">&lt;</span><span class="nt">h2</span><span class="p">&gt;</span>Organization<span class="p">&lt;/</span><span class="nt">h2</span><span class="p">&gt;</span> <span class="p">&lt;</span><span class="nt">div</span> <span class="na">class</span><span class="o">=</span><span class="s">&quot;form-row&quot;</span><span class="p">&gt;</span> <span class="p">&lt;</span><span class="nt">a</span> <span class="na">href</span><span class="o">=</span><span class="s">&quot;</span><span class="cp">{%</span> <span class="k">url</span> <span class="s1">&#39;admin:organizations_organization_change&#39;</span> <span class="nv">original.editor.organization.pk</span> <span class="cp">%}</span><span class="s">&quot;</span><span class="p">&gt;</span><span class="cp">{{</span> <span class="nv">original.editor.organization</span> <span class="cp">}}</span><span class="p">&lt;/</span><span class="nt">a</span><span class="p">&gt;</span> <span class="p">&lt;/</span><span class="nt">div</span><span class="p">&gt;</span> <span class="p">&lt;/</span><span class="nt">fieldset</span><span class="p">&gt;</span> <span class="cp">{%</span> <span class="k">endif</span> <span class="cp">%}</span> <span class="cp">{%</span> <span class="k">endblock</span> <span class="nv">after_related_objects</span> <span class="cp">%}</span> </code></pre></div> <p>The <code>original</code> context variable contains the object being edited. The <code>editor</code> attribute is the reverse accessor for the <code>OneToOneField</code> mentioned above.</p> <h2 id="method-2-add-a-method-to-the-model-admin-class-returning-a-html-blob"><a class="toclink" href="#method-2-add-a-method-to-the-model-admin-class-returning-a-html-blob">Method 2: Add a method to the model admin class returning a HTML blob</a></h2> <p>A terrible but also nice way is to add a method to the <code>ModelAdmin</code> class which returns the HTML containing the links you want, and adding the name of the method to <code>readonly_fields</code>. This is even mentioned in <a href="https://docs.djangoproject.com/en/stable/ref/contrib/admin/#django.contrib.admin.ModelAdmin.readonly_fields">the official <code>readonly_fields</code> documentation</a> but I discovered this by accident a few years back.</p> <p>The method name doesn&rsquo;t have to be added anywhere else, not to <code>fields</code> nor do you have to define <code>fieldsets</code> for this to work. Just adding it to <code>readonly_fields</code> appends it to the end of the form, before any eventual inlines you&rsquo;re using.</p> <div class="chl"><pre><span></span><code><span class="kn">from</span> <span class="nn">django.template.loader</span> <span class="kn">import</span> <span class="n">render_to_string</span> <span class="kn">from</span> <span class="nn">app</span> <span class="kn">import</span> <span class="n">models</span> <span class="nd">@admin</span><span class="o">.</span><span class="n">register</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">Class</span><span class="p">)</span> <span class="k">class</span> <span class="nc">ClassAdmin</span><span class="p">(</span><span class="n">admin</span><span class="o">.</span><span class="n">ModelAdmin</span><span class="p">):</span> <span class="n">list_display</span> <span class="o">=</span> <span class="p">[</span><span class="s2">&quot;name&quot;</span><span class="p">,</span> <span class="s2">&quot;language_code&quot;</span><span class="p">]</span> <span class="n">readonly_fields</span> <span class="o">=</span> <span class="p">[</span><span class="s2">&quot;admin_show_custom_districts&quot;</span><span class="p">]</span> <span class="nd">@admin</span><span class="o">.</span><span class="n">display</span><span class="p">(</span><span class="n">description</span><span class="o">=</span><span class="n">_</span><span class="p">(</span><span class="s2">&quot;districts&quot;</span><span class="p">)</span> <span class="k">def</span> <span class="nf">admin_show_custom_districts</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">obj</span><span class="p">):</span> <span class="k">return</span> <span class="n">render_to_string</span><span class="p">(</span> <span class="s2">&quot;admin/admin_show_custom_districts.html&quot;</span><span class="p">,</span> <span class="p">{</span><span class="s2">&quot;custom_districts&quot;</span><span class="p">:</span> <span class="n">obj</span><span class="o">.</span><span class="n">customdistrict_set</span><span class="o">.</span><span class="n">all</span><span class="p">()},</span> <span class="p">)</span> </code></pre></div>Weeknotes (2025 week 03)https://406.ch/writing/weeknotes-2025-week-03/2025-01-15T12:00:00Z2025-01-15T12:00:00Z<h1 id="weeknotes-2025-week-03"><a class="toclink" href="#weeknotes-2025-week-03">Weeknotes (2025 week 03)</a></h1> <h2 id="claude-ai-helped-me-for-the-first-time"><a class="toclink" href="#claude-ai-helped-me-for-the-first-time">Claude AI helped me for the first time</a></h2> <p><a href="https://github.com/matthiask/django-imagefield">django-imagefield</a> prefers processing thumbnails, cropped images etc. directly when saving the model and not later on demand; it&rsquo;s faster and also you&rsquo;ll know it immediately when an image couldn&rsquo;t be processed for some reason instead of only later when people actually try browsing your site.</p> <p>A consequence is that if you change formats you have to remember that you have to reprocess the images. The Django app comes with a management command <code>./manage.py process_imagefields</code> to help with this. I have added parallel processing based on <code>concurrent.futures</code> to it some time ago so that the command completes faster when it is being run on a system with several cores.</p> <p>A work colleague is using macOS (many are, in fact), and he always got multiprocessing Python crashes. This is a well known issue and I remember reading about it a few years ago. I checked the docs and saw that the <a href="https://docs.python.org/3/library/concurrent.futures.html"><code>concurrent.futures</code></a> page doesn&rsquo;t mention macOS, but <a href="https://docs.python.org/3/library/multiprocessing.html"><code>multiprocessing</code></a> does. So, I hoped that a simple rewrite of the management command using <code>multiprocessing</code> might fix it.</p> <p>Because I was in a rush and really didn&rsquo;t want to do it I turned to an AI assistant for doing this boring work. To my surprise it immediately produced a version which I could easily fix by hand to produce a working version. Of course, the initial response was totally broken, removed code it wasn&rsquo;t supposed to, and even the syntax was invalid. I didn&rsquo;t expect more though, but what was surprising was that it actually felt like I had to do less work at this time.</p> <p>The assistant also helped adding a <code>--no-parallel</code> flag to the management command. The output was even more broken than the output of the change mentioned above, but again, I could easily fix it to achieve what I wanted.</p> <p>The fact that I know the code and <a href="https://git-scm.com/">git</a> well certainly helped, the assistant would really have helped without that knowledge.</p> <p>In the end, switching to <code>multiprocessing</code> didn&rsquo;t help, but adding the <code>--no-parallel</code> flag allowed them to run the processing themselves by not spawning any additional threads or processes.</p> <p>The energy use and the stealing of copyrighted material done by the AI companies is still really bad. It does feel somewhat OK to use an AI assistant in an area where I&rsquo;m proficient as well and where I probably also supplied training material (without being asked if I wanted this) though. It&rsquo;s making me slightly faster, and doesn&rsquo;t allow me to do things I really couldn&rsquo;t otherwise.</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/FeinCMS/">FeinCMS 24.12.3</a>: I have added a TinyMCE 7 integration to FeinCMS.</li> <li><a href="https://pypi.org/project/django-imagefield/">django-imagefield 0.21.1</a>: See above.</li> </ul>Weeknotes (2024 week 51)https://406.ch/writing/weeknotes-2024-week-51/2024-12-20T12:00:00Z2024-12-20T12:00:00Z<h1 id="weeknotes-2024-week-51"><a class="toclink" href="#weeknotes-2024-week-51">Weeknotes (2024 week 51)</a></h1> <h2 id="building-forms-using-django"><a class="toclink" href="#building-forms-using-django">Building forms using Django</a></h2> <p>I last wrote about this topic <a href="https://406.ch/writing/building-forms-with-the-django-admin/">in April</a>. It has <a href="https://mastodon.social/@webology/113669270531953652">resurfaced on Mastodon this week</a>. I&rsquo;m thinking about writing a <a href="https://github.com/feincms/feincms3-forms">feincms3-forms</a> demo app, but I already have too much on my plate. I think composing a forms builder on top of <a href="https://django-content-editor.readthedocs.io/">django-content-editor</a> is the way to go, instead of replacing the admin interface altogether &ndash; sure, you can always do that, but it&rsquo;s so much less composable&hellip;</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/blacknoise/">blacknoise 1.2</a>: No real changes, added support for Python 3.13 basically without changing anything. It&rsquo;s always nice when this happens.</li> <li><a href="https://pypi.org/project/django-imagefield/">django-imagefield 0.21</a></li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.10</a>: I rebuilt django-prose-editor from the ground up <a href="https://406.ch/writing/rebuilding-django-prose-editor-from-the-ground-up/">and wrote about that two weeks ago</a>. The 0.10 release marks the final point of this particular rewrite.</li> <li><a href="https://pypi.org/project/django-js-asset/">django-js-asset 3.0</a>: See the blog post from <a href="https://406.ch/writing/object-based-assets-for-django-s-forms-media/">this week</a></li> </ul>Object-based assets for Django's forms.Mediahttps://406.ch/writing/object-based-assets-for-django-s-forms-media/2024-12-18T12:00:00Z2024-12-18T12:00:00Z<h1 id="object-based-assets-for-djangos-formsmedia"><a class="toclink" href="#object-based-assets-for-djangos-formsmedia">Object-based assets for Django&rsquo;s forms.Media</a></h1> <p>The pull request for adding <a href="https://github.com/django/django/pull/18782">object-based script media assets into Django</a> is in a good state and I hope it will be merged soon. I have been using object-based assets long before <a href="https://github.com/django/django/commit/4c76ffc2d6c77">Django actually added support for them in 4.1</a> (<a href="https://github.com/feincms/django-content-editor/commit/82ac91ea7af2409bb3672e11c18871002ddc9753">since 2016</a>, that&rsquo;s before Django 1.10!) by using a gross hack. Luckily I have been able to clean up the code when Django 4.1 landed.</p> <p>I have been asking myself at times why I haven&rsquo;t proposed the change to Django myself despite having been a user of something like this for such a long time. After all, I have been happily contributing issue reports, bug fixes and tests to Django. The process of adding new features sometimes is terribly frustrating though even when looking (and cheering) from the sidelines. It feels bad that adding another package to the <a href="https://pypi.org/user/matthiask/">list of packages I maintain</a> so clearly seems to be the better way to <strong>get things done</strong> compared to proposing a new feature for Django itself. I hope <a href="https://406.ch/writing/weeknotes-2024-week-49/">processes change somewhat</a>.</p> <p>But I digress.</p> <p>The <code>ProseEditorWidget</code> in <a href="https://github.com/matthiask/django-prose-editor/">django-prose-editor</a> wants to ship CSS, JavaScript and some JSON to the browser for the widget. So, of course I used object-based media assets for this instead of widget HTML templates. Media assets are deduplicated and sorted by Django. If different editor presets use differing lists of assets they are smartly merged by <code>forms.Media</code> using a topological sort. You get those niceties for free when using <code>forms.Media</code> and everything just works, so what&rsquo;s not to like?</p> <p>The only thing which isn&rsquo;t to like is that Django, at the time of writing, doesn&rsquo;t provide any classes helping with this. You can put strings into <code>forms.Media</code> or you can put objects with a <code>__html__()</code> method in there. The latter of course is all that&rsquo;s needed to support more advanced use cases &ndash; and that&rsquo;s exactly what <a href="https://pypi.org/project/django-js-asset/">django-js-asset</a> now provides, and what django-prose-editor uses.</p> <p><a href="https://pypi.org/project/django-js-asset/">django-js-asset</a> has long supported a <code>JS</code> class with support for additional attributes, for example:</p> <div class="chl"><pre><span></span><code><span class="kn">from</span> <span class="nn">js_asset</span> <span class="kn">import</span> <span class="n">JS</span> <span class="n">forms</span><span class="o">.</span><span class="n">Media</span><span class="p">(</span><span class="n">js</span><span class="o">=</span><span class="p">[</span> <span class="n">JS</span><span class="p">(</span><span class="s2">&quot;asset.js&quot;</span><span class="p">,</span> <span class="p">{</span><span class="s2">&quot;id&quot;</span><span class="p">:</span> <span class="s2">&quot;asset-script&quot;</span><span class="p">,</span> <span class="s2">&quot;data-answer&quot;</span><span class="p">:</span> <span class="s2">&quot;42&quot;</span><span class="p">}),</span> <span class="p">])</span> </code></pre></div> <p>Since 3.0 the package also comes with a <code>CSS</code> and <code>JSON</code> class:</p> <div class="chl"><pre><span></span><code><span class="kn">from</span> <span class="nn">js_asset</span> <span class="kn">import</span> <span class="n">CSS</span><span class="p">,</span> <span class="n">JS</span><span class="p">,</span> <span class="n">JSON</span> <span class="n">forms</span><span class="o">.</span><span class="n">Media</span><span class="p">(</span><span class="n">js</span><span class="o">=</span><span class="p">[</span> <span class="n">JSON</span><span class="p">({</span><span class="s2">&quot;cfg&quot;</span><span class="p">:</span> <span class="mi">42</span><span class="p">},</span> <span class="nb">id</span><span class="o">=</span><span class="s2">&quot;widget-cfg&quot;</span><span class="p">),</span> <span class="n">CSS</span><span class="p">(</span><span class="s2">&quot;widget/style.css&quot;</span><span class="p">),</span> <span class="n">CSS</span><span class="p">(</span><span class="s2">&quot;p{color:red;}&quot;</span><span class="p">,</span> <span class="n">inline</span><span class="o">=</span><span class="kc">True</span><span class="p">),</span> <span class="n">JS</span><span class="p">(</span><span class="s2">&quot;widget/script.js&quot;</span><span class="p">,</span> <span class="p">{</span><span class="s2">&quot;type&quot;</span><span class="p">:</span> <span class="s2">&quot;module&quot;</span><span class="p">}),</span> <span class="p">])</span> </code></pre></div> <p>This produces the following HTML:</p> <div class="chl"><pre><span></span><code><span class="p">&lt;</span><span class="nt">script</span> <span class="na">id</span><span class="o">=</span><span class="s">&quot;widget-cfg&quot;</span> <span class="na">type</span><span class="o">=</span><span class="s">&quot;application/json&quot;</span><span class="p">&gt;{</span><span class="s2">&quot;cfg&quot;</span><span class="o">:</span><span class="w"> </span><span class="mf">42</span><span class="p">}&lt;/</span><span class="nt">script</span><span class="p">&gt;</span> <span class="p">&lt;</span><span class="nt">link</span> <span class="na">rel</span><span class="o">=</span><span class="s">&quot;stylesheet&quot;</span> <span class="na">href</span><span class="o">=</span><span class="s">&quot;/static/widget/style.css&quot;</span><span class="p">&gt;</span> <span class="p">&lt;</span><span class="nt">style</span><span class="p">&gt;</span><span class="nt">p</span><span class="p">{</span><span class="k">color</span><span class="p">:</span><span class="kc">red</span><span class="p">;}&lt;/</span><span class="nt">style</span><span class="p">&gt;</span> <span class="p">&lt;</span><span class="nt">script</span> <span class="na">src</span><span class="o">=</span><span class="s">&quot;/static/widget/script.js&quot;</span> <span class="na">type</span><span class="o">=</span><span class="s">&quot;module&quot;</span><span class="p">&gt;&lt;/</span><span class="nt">script</span><span class="p">&gt;</span> </code></pre></div> <p>The code which is proposed for Django supports the JavaScript use case but with a slightly different API:</p> <div class="chl"><pre><span></span><code><span class="kn">from</span> <span class="nn">django.forms</span> <span class="kn">import</span> <span class="n">Script</span> <span class="n">forms</span><span class="o">.</span><span class="n">Media</span><span class="p">(</span><span class="n">js</span><span class="o">=</span><span class="p">[</span> <span class="n">Script</span><span class="p">(</span><span class="s2">&quot;widget/script.js&quot;</span><span class="p">,</span> <span class="nb">type</span><span class="o">=</span><span class="s2">&quot;module&quot;</span><span class="p">),</span> <span class="p">])</span> </code></pre></div> <p>This looks slightly nicer as long as you don&rsquo;t use e.g. data attributes, because then you have to do:</p> <div class="chl"><pre><span></span><code><span class="kn">from</span> <span class="nn">django.forms</span> <span class="kn">import</span> <span class="n">Script</span> <span class="n">forms</span><span class="o">.</span><span class="n">Media</span><span class="p">(</span><span class="n">js</span><span class="o">=</span><span class="p">[</span> <span class="n">Script</span><span class="p">(</span><span class="s2">&quot;widget/script.js&quot;</span><span class="p">,</span> <span class="o">**</span><span class="p">{</span><span class="s2">&quot;data-cfg&quot;</span><span class="p">:</span> <span class="o">...</span><span class="p">}),</span> <span class="p">])</span> </code></pre></div> <p>I always forget that Python supports passing keyword arguments names which aren&rsquo;t valid Python identifiers (but only when using <code>**kwargs</code>). I personally don&rsquo;t care much either way, and when my packages can finally drop compatibility with Django versions which do not support all these functionalities yet I&rsquo;ll finally be able to retire <a href="https://pypi.org/project/django-js-asset/">django-js-asset</a>. That won&rsquo;t happen any time soon though, if only because I like supporting old versions of Django because I have so many Django-based websites running somewhere.</p>Weeknotes (2024 week 49)https://406.ch/writing/weeknotes-2024-week-49/2024-12-06T12:00:00Z2024-12-06T12:00:00Z<h1 id="weeknotes-2024-week-49"><a class="toclink" href="#weeknotes-2024-week-49">Weeknotes (2024 week 49)</a></h1> <h2 id="django-steering-council-elections"><a class="toclink" href="#django-steering-council-elections">Django Steering Council elections</a></h2> <p>I have been thinking long and hard about running for the Django Steering Council. I think there are a few things I could contribute since I&rsquo;ve been using Django for 16 or more years, and have been working on, maintaining and publishing third-party apps almost all this time. I have also contributed a few small features to Django core itself, and contributed my fair share of tests and bugfixes. The reason why I haven&rsquo;t been more involved was always that I feared the review process with what I perceive to be a too unrestrained perfectionism. Teaching people is good, but I fear that those who teach are self-selected survivors of the process, which come to appreciate the perfectionism a bit too much. It&rsquo;s somewhat the same as with the Swiss naturalization process &ndash; the hurdles are very high, and some of those who weather the process maybe are or grow to be too fond of it.</p> <p>An important point is that this has nothing to do with being nice (or not). Everybody has always been great, maybe with the exception of myself back when I didn&rsquo;t understand that the problem wasn&rsquo;t the individuals but the way everyone has agreed things should be done.</p> <p>I&rsquo;m not the only one who thinks that we <a href="https://knowyourmeme.com/memes/we-should-improve-society-somewhat">should improve the process somewhat</a>. So, I&rsquo;m definitely going to look out for candidates who think this is important.</p> <p>There are a few reasons why I&rsquo;m not running myself at this time. A somewhat important reason is that my candidacy wouldn&rsquo;t help diversity at all. This shouldn&rsquo;t discourage anyone else with the same background from running &ndash; we cannot change the world all at once. More importantly, I have more personal reasons for being hesitant to accept new commitments. That being said, I&rsquo;m looking forward to be more involved in the community in other ways. And also, it&rsquo;s not now or never.</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/feincms3-cookiecontrol/">feincms3-cookiecontrol 1.5.4</a>: No functional changes, only code golfing. It&rsquo;s nice to have a working cookie banner with a solution for embedding third party content only when people consent in less than 4KiB of minified (not compressed!) JavaScript.</li> <li><a href="https://pypi.org/project/django-admin-ordering/">django-admin-ordering 0.20</a>: Objects can now be reordered using arrow buttons instead of drag drop or manually changing the ordering field&rsquo;s value. This should make the package more accessible. It&rsquo;s always a joy when people contribute such useful improvements.</li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.10a?</a>: See <a href="https://406.ch/writing/rebuilding-django-prose-editor-from-the-ground-up/">the recent blog post</a>.</li> </ul>Rebuilding django-prose-editor from the ground uphttps://406.ch/writing/rebuilding-django-prose-editor-from-the-ground-up/2024-12-04T12:00:00Z2024-12-04T12:00:00Z<h1 id="rebuilding-django-prose-editor-from-the-ground-up"><a class="toclink" href="#rebuilding-django-prose-editor-from-the-ground-up">Rebuilding django-prose-editor from the ground up</a></h1> <p>The <a href="https://pypi.org/project/django-prose-editor/">django-prose-editor</a> package provides a HTML editor based upon the <a href="https://prosemirror.net/">ProseMirror toolkit</a> for the Django administration interface and for the frontend.</p> <p>The package has been extracted from a customer project and open sourced so that it could be used in other projects as well. It followed a very restricted view of how rich text editors should work, which I have initially added to the <a href="https://github.com/feincms/feincms/commit/70cd7a1244438d2ba97852256f77daa2c870c345#diff-556c5559a716059d4fb714ad34de6a9845870e8d55bbd2cb9d77c732eb961388">FeinCMS repository when documenting the design decisions more than 15 years ago</a> <small>(Note that I didn&rsquo;t edit the paragraph, it&rsquo;s reproduced here as it was back then, with all the errors and heedlessness.)</small></p> <blockquote> <p>All of this convinced me that offering the user a rich text editor with too much capabilites is a really bad idea. The rich text editor in FeinCMS only has bold, italic, bullets, link and headlines activated (and the HTML code button, because that&rsquo;s sort of inevitable &ndash; sometimes the rich text editor messes up and you cannot fix it other than going directly into the HTML code. Plus, if someone really knows what he&rsquo;s doing, I&rsquo;d still like to give him the power to shot his own foot).</p> </blockquote> <p>My personal views are unchanged. I have to recognize though that forcing this idea upon everyone isn&rsquo;t workable and that this would mean that I&rsquo;d have to find a different editor for most projects just because people really want or need more rope. Going back to an editor which allows everything was out of the question, so I had to look around for a way to allow project-specific extensions for the editor.</p> <p>Of course that&rsquo;s problematic, since Django packages and Python virtualenvs do not offer a good way of shipping CSS and JavaScript which should be available for a frontend bundler to process. The existing <a href="https://docs.djangoproject.com/en/5.1/ref/contrib/staticfiles/">Django staticfiles app</a> is great, works well, but it&rsquo;s not a bundler &ndash; and it shouldn&rsquo;t be.</p> <p>So, I started shopping around for ways to make ProseMirror extensible while keeping extensions clean and well localized. Instead of inventing another plugin ecosystem I settled on <a href="https://tiptap.dev/">Tiptap</a> which uses ProseMirror under the hood. The abstractions are pleasantly leaky &ndash; if you know how to work with ProseMirror&rsquo;s API, you can use Tiptap&rsquo;s API without any issues. That was important for me, since I already have a somewhat large selection of plugins which I do not want to reimplement from the ground up.</p> <p>I had already looked at Tiptap a few years back, but ultimately stayed with ProseMirror because I liked some behaviors better (such as not including trailing spaces in marks) and because I didn&rsquo;t need the extensibility which at the time only made the resulting bundle much bigger.</p> <p>Now, things have improved a lot, and I&rsquo;m really happy with Tiptap and the development version of django-prose-editor. <a href="https://github.com/matthiask/django-prose-editor/?tab=readme-ov-file#customization">Writing an editor extension in project code is great</a>, and my editor core stays nice. Also the list of readily available extensions is large, and most of the things just work.</p>Weeknotes (2024 week 47)https://406.ch/writing/weeknotes-2024-week-47/2024-11-20T12:00:00Z2024-11-20T12:00:00Z<h1 id="weeknotes-2024-week-47"><a class="toclink" href="#weeknotes-2024-week-47">Weeknotes (2024 week 47)</a></h1> <p>I missed a single co-writing session and of course that lead to four weeks of no posts at all to the blog. Oh well.</p> <h2 id="debugging"><a class="toclink" href="#debugging">Debugging</a></h2> <p>I want to share a few debugging stories from the last weeks.</p> <h3 id="pillow-11-and-djangos-get_image_dimensions"><a class="toclink" href="#pillow-11-and-djangos-get_image_dimensions">Pillow 11 and Django&rsquo;s <code>get_image_dimensions</code></a></h3> <p>The goal of <a href="https://github.com/matthiask/django-imagefield">django-imagefield</a> was to deeply verify that Django and Pillow are able to work with uploaded files; some files can be loaded, their dimensions can be inspected, but problems happen later when Pillow actually tries resizing or filtering files. Because of this django-imagefield does more work when images are added to the system instead of working around it later. (Django doesn&rsquo;t do this on purpose because doing all this work up-front could be considered a DoS factor.)</p> <p>In the last weeks I suddenly got recurring errors from saved files again, something which shouldn&rsquo;t happen, but obviously did.</p> <p>Django wants to read image dimensions when accessing or saving image files (by the way, always use <code>height_field</code> and <code>width_field</code>, otherwise Django will open and inspect image files even when you&rsquo;re only loading Django models from the database&hellip;!) and it uses a smart and wonderful<sup id="fnref:fn1"><a class="footnote-ref" href="#fn:fn1">1</a></sup> hack to do this: It reads a few hundred bytes from the image file, instructs Pillow to inspect the file and if an exception happens it reads more bytes and tries again. This process relies on the exact type of exceptions raised internally though, and the release of Pillow 11 changed the types&hellip; for some file types only. Fun times.</p> <p>The issue had already been reported as <a href="https://code.djangoproject.com/ticket/33240">#33240</a> and is now tracked as <a href="https://github.com/python-pillow/Pillow/issues/8530">#8530</a> on the Pillow issue tracker. Let&rsquo;s see what happens. For now, django-imagefield declares itself to be incompatible with Pillow 11.0.0 so that this error cannot happen.</p> <h3 id="rspack-and-lightningcss-shuffled-css-properties"><a class="toclink" href="#rspack-and-lightningcss-shuffled-css-properties">rspack and lightningcss shuffled CSS properties</a></h3> <p><a href="https://rspack.dev/">rspack</a> 1.0 started reordering CSS properties which of course lead to CSS properties overriding each other in the incorrect order. That was a fun one to debug. I tracked the issue down to the switch from the swc CSS minimizer to <a href="https://github.com/parcel-bundler/lightningcss">lightningcss</a> and submitted a reproduction to the <a href="https://github.com/parcel-bundler/lightningcss/issues/805#issuecomment-2358219597">issue tracker</a>. My rust knowledge wasn&rsquo;t up to the task of attempting to submit a fix myself. Luckily, it has been fixed in the meantime.</p> <h3 id="rspack-problems"><a class="toclink" href="#rspack-problems">rspack problems</a></h3> <p>I have another problem with rspack where I haven&rsquo;t yet tracked down the issue. rspack produces a broken bundle starting with <a href="https://github.com/web-infra-dev/rspack/releases/tag/v1.0.0-beta.2">1.0.0-beta.2</a> when compiling a particular project of mine. I have the suspicion that I have misconfigured some stuff related to import paths and yarn workspaces. I have no idea how anyone could have a complete understanding of these things&hellip;</p> <p><strong>Update:</strong> The problem was <a href="https://github.com/web-infra-dev/rspack/issues/8027">#8027</a>, <code>experiments.css</code> is quite broken for now.</p> <p>Bundlers are complex beasts, and I&rsquo;m happy that I mostly can just use them.</p> <h3 id="closing-thoughts"><a class="toclink" href="#closing-thoughts">Closing thoughts</a></h3> <p>Debugging is definitely a rewarding activity for me. I like tracking stuff down like this. Unfortunately, problems always tend to crop up when time is scarce already, but what can you do.</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <p>Quite a few releases, many of them verifying Python 3.13 and Django 5.1 support (if it hasn&rsquo;t been added already in previous releases). The nicest part: If I remember correctly I didn&rsquo;t have to change anything anywhere, everything just continues to work.</p> <ul> <li><a href="https://pypi.org/project/django-admin-ordering/">django-admin-ordering 0.19</a>: I added support for automatically renumbering objects on page load. This is mostly useful if you already have existing data which isn&rsquo;t ordered yet.</li> <li><a href="https://pypi.org/project/feincms3-data/">feincms3-data 0.7</a>: Made sure that objects are dumped in a deterministic order when dumping. I wanted to compare JSON dumps by hand before and after a big data migration in a customer project and differently ordered dumps made the comparison impossible.</li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.9</a>: I updated the ProseMirror packages, and put the editor into read-only mode for <code>&lt;textarea disabled&gt;</code> elements.</li> <li><a href="https://pypi.org/project/feincms3-language-sites/">feincms3-language-sites 0.4</a>: Finally released the update containing the necessary hook to validate page trees and their unique paths before moving produces integrity errors. Error messages are nicer than internal server errors.</li> <li><a href="https://pypi.org/project/django-authlib/">django-authlib 0.17.2</a>: The value of the cookie which is used to save the URL where users should be redirected to after authentication wasn&rsquo;t checked for validity when setting it, only when reading it. This meant that attackers could produce invalid header errors in application servers. No real security problem here when using authlib&rsquo;s code.</li> <li><a href="https://pypi.org/project/feincms3/">feincms3 5.3</a>: Minor update which mostly removes support for outdated Python and Django versions.</li> <li><a href="https://pypi.org/project/django-imagefield/">django-imagefield 0.20</a>: See above.</li> </ul> <div class="footnote"> <hr /> <ol> <li id="fn:fn1"> <p>wonderfully ugly&#160;<a class="footnote-backref" href="#fnref:fn1" title="Jump back to footnote 1 in the text">&#8617;</a></p> </li> </ol> </div>Weeknotes (2024 week 43)https://406.ch/writing/weeknotes-2024-week-43/2024-10-23T12:00:00Z2024-10-23T12:00:00Z<h1 id="weeknotes-2024-week-43"><a class="toclink" href="#weeknotes-2024-week-43">Weeknotes (2024 week 43)</a></h1> <p>I had some much needed time off, so this post isn&rsquo;t that long even though <a href="https://406.ch/writing/weeknotes-2024-week-39/">four weeks have passed since the last entry</a>.</p> <h2 id="from-webpack-to-rspack"><a class="toclink" href="#from-webpack-to-rspack">From webpack to rspack</a></h2> <p>I&rsquo;ve been really happy with <a href="https://rspack.dev/">rspack</a> lately. Converting webpack projects to rspack is straightforward since it mostly supports the same configuration, but it&rsquo;s much much faster since it&rsquo;s written in Rust. Rewriting things in Rust is a recurring theme, but in this case it really helps a lot. Building the frontend of a larger project of ours consisting of several admin tools and complete frontend implementations for different teaching materials only takes 10 seconds now instead of several minutes. That&rsquo;s a big and relevant difference.</p> <p>Newcomers should probably still either use <a href="https://rsbuild.dev/">rsbuild</a>, <a href="https://vite.dev/">Vite</a> or maybe no bundler at all. Vanilla JS and browser support for ES modules is great. That being said, I like cache busting, optimized bundling and far-future expiry headers in production and hot module reloading in development a lot, so learning to work with a frontend bundler is definitely still worth it.</p> <h2 id="dark-mode-and-light-mode"><a class="toclink" href="#dark-mode-and-light-mode">Dark mode and light mode</a></h2> <p>I have been switching themes in my preferred a few times per year in the past. The following ugly bit of vimscript helps switch me the theme each time the sun comes out when working outside:</p> <div class="chl"><pre><span></span><code><span class="nv">let</span><span class="w"> </span><span class="nv">t</span>:<span class="nv">light</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="mi">0</span> <span class="nv">function</span><span class="o">!</span><span class="w"> </span><span class="nv">FiatLux</span><span class="ss">()</span> <span class="w"> </span><span class="k">if</span><span class="w"> </span><span class="nv">t</span>:<span class="nv">light</span><span class="w"> </span><span class="o">==</span><span class="w"> </span><span class="mi">0</span> <span class="w"> </span>:<span class="nv">set</span><span class="w"> </span><span class="nv">background</span><span class="o">=</span><span class="nv">light</span> <span class="w"> </span><span class="nv">let</span><span class="w"> </span><span class="nv">t</span>:<span class="nv">light</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="mi">1</span> <span class="w"> </span><span class="k">else</span> <span class="w"> </span>:<span class="nv">set</span><span class="w"> </span><span class="nv">background</span><span class="o">=</span><span class="nv">dark</span> <span class="w"> </span><span class="nv">let</span><span class="w"> </span><span class="nv">t</span>:<span class="nv">light</span><span class="w"> </span><span class="o">=</span><span class="w"> </span><span class="mi">0</span> <span class="w"> </span><span class="k">endif</span> <span class="nv">endfunction</span> <span class="nv">nnoremap</span><span class="w"> </span><span class="o">&lt;</span><span class="nv">F12</span><span class="o">&gt;</span><span class="w"> </span>:<span class="k">call</span><span class="w"> </span><span class="nl">FiatLux</span><span class="ss">()</span><span class="o">&lt;</span><span class="nv">CR</span><span class="o">&gt;</span> </code></pre></div> <p>I&rsquo;m using the <a href="https://devsuite.app/ptyxis/">Ptyxis</a> terminal emulator currently, I haven&rsquo;t investigated yet if there&rsquo;s a shortcut to toggle dark and light mode for it as well. Using F10 to open the main menu works fine though, and using the mouse wouldn&rsquo;t be painful either.</p> <h2 id="helping-out-in-the-django-forum-and-the-discord"><a class="toclink" href="#helping-out-in-the-django-forum-and-the-discord">Helping out in the Django forum and the Discord</a></h2> <p>I have found some pleasure in helping out in the <a href="https://forum.djangoproject.com/">Django Forum</a> and in the official <a href="https://discord.gg/xcRH6mN4fa">Django Discord</a>. I sometimes wonder why more people aren&rsquo;t reading the Django source code when they hit something which looks like a bug or something which they do not understand. I find Django&rsquo;s source code very readable and I have found many nuggets within it. I&rsquo;d always recommend checking the documentation or maybe official help channels first, but the code is also out there and that fact should be taken advantage of.</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/django-content-editor/">django-content-editor 7.1</a>: Fixed a bug where the ordering and region fields were handled incorrectly when they appear on one line in the fieldset. Also improved the presentation of inlines in unknown regions and clarified the meaning of the move to region dropdown. Also, released the improvements from previous patch releases as a new minor release because that&rsquo;s what I should have been doing all along.</li> <li><a href="https://pypi.org/project/form-designer/">form-designer 0.27</a>: A user has been bitten by <code>slugify</code> removing cyrillic characters because it only keeps ASCII characters around. Here&rsquo;s the wontfixed bug in the Django issue tracker: <a href="https://code.djangoproject.com/ticket/8391">#8391</a>. I fixed the issue by removing the slugification (is that even a word?) when generating choices.</li> </ul>Weeknotes (2024 week 39)https://406.ch/writing/weeknotes-2024-week-39/2024-09-25T12:00:00Z2024-09-25T12:00:00Z<h1 id="weeknotes-2024-week-39"><a class="toclink" href="#weeknotes-2024-week-39">Weeknotes (2024 week 39)</a></h1> <h2 id="css-for-django-forms"><a class="toclink" href="#css-for-django-forms">CSS for Django forms</a></h2> <p>Not much going on in OSS land. I have been somewhat active in the official Django forum, discussing ways to add Python-level hooks to allow adding CSS classes around form fields and their labels. The discussion on the <a href="https://forum.djangoproject.com/t/proposal-make-it-easy-to-add-css-classes-to-a-boundfield/32022">forum</a> and on the <a href="https://github.com/django/django/pull/18266">pull request</a> goes in the direction of allowing using custom <code>BoundField</code> classes per form or even per project (instead of only per field as is already possible today). This would allow overriding <code>css_classes</code>, e.g. to add a simple <code>class="field"</code>. Together with <code>:has()</code> this would probably allow me to skip using custom HTML templates in 99% of all cases.</p> <p>I have also been lurking in the Discord, but more to help and less to promote my packages and ideas :-)</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/django_user_messages/">django-user-messages 1.1</a>: Added Django 5.0, 5.1 to the CI, and fixed the migrations to no longer mention <code>index_together</code> at all. It seems that squashing the migrations wasn&rsquo;t sufficient, I also had to actually delete the old migrations.</li> <li><a href="https://pypi.org/project/blacknoise/">blacknoise 1.1</a>: <a href="https://www.starlette.io/">Starlette</a>&rsquo;s <code>FileResponse</code> has gained support for the HTTP Range header, allowing me to remove my homegrown implementation from the package. The blacknoise implementation is now half as long as it was in 1.0.</li> <li><a href="https://pypi.org/project/django_fhadmin/">django-fhadmin 2.3</a>: No new features, only tweaks to the styling and behavior prompted by updates to Django&rsquo;s admin interface.</li> <li><a href="https://pypi.org/project/django-cabinet/">django-cabinet 0.17</a>: I have pruned the CI matrix and accepted a pull request adding a ru translation. I feel conflicted about that since I strongly believe that everything is political, but I don&rsquo;t know if rejecting translations helps anyone.</li> </ul>django-content-editor now supports nested sectionshttps://406.ch/writing/django-content-editor-now-supports-nested-sections/2024-09-13T12:00:00Z2024-09-13T12:00:00Z<h1 id="django-content-editor-now-supports-nested-sections"><a class="toclink" href="#django-content-editor-now-supports-nested-sections">django-content-editor now supports nested sections</a></h1> <p><a href="https://django-content-editor.readthedocs.io/">django-content-editor</a> (and it&rsquo;s ancestor FeinCMS) has been the Django admin extension for editing content consisting of reusable blocks since 2009. In the last years we have more and more often started <a href="https://feincms3.readthedocs.io/en/latest/guides/rendering.html#grouping-plugins-into-subregions">automatically grouping related items</a>, e.g. for rendering a sequence of images as a gallery. But, sometimes it&rsquo;s nice to give editors more control. This has been possible by using blocks which open a subsection and blocks which close a subsection for a long time, but it hasn&rsquo;t been friendly to content managers, especially when using nested sections.</p> <p>The content editor now has first-class support for such nested sections. Here&rsquo;s a screenshot showing the nesting:</p> <p><img alt="django-content-editor with sections" src="https://406.ch/assets/20240911-content-editor-sections.png" /></p> <p>Finally it&rsquo;s possible to visually group blocks into sections, collapse those sections as once and drag and drop whole sections into their place instead of having to select the involved blocks individually.</p> <p>The best part about it is that the content editor still supports all Django admin widgets, as long as those widgets have support for the Django administration interface&rsquo;s <a href="https://docs.djangoproject.com/en/latest/ref/contrib/admin/javascript/">inline form events</a>! Moving DOM nodes around breaks attached JavaScript behaviors, but we do not actually move DOM nodes around after the initialization &ndash; instead, we use <a href="https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_flexible_box_layout/Ordering_flex_items">Flexbox ordering</a> to visually reorder blocks. It&rsquo;s a bit more work than using a ready-made sortable plugin, but &ndash; as mentioned &ndash; the prize is that we don&rsquo;t break any other Django admin extensions.</p> <h2 id="simple-patterns"><a class="toclink" href="#simple-patterns">Simple patterns</a></h2> <p>I previously already reacted to a blog post by Lincoln Loop here in my post <a href="https://406.ch/writing/my-reaction-to-the-block-driven-cms-blog-post/">My reaction to the block-driven CMS blog post</a>.</p> <p>The latest blog post, <a href="https://lincolnloop.com/insights/simple-block-pattern-wagtail-cms/">Solving the Messy Middle: a Simple Block Pattern for Wagtail CMS</a> was interesting as well. It dives into the configuration of a <a href="https://wagtail.org/">Wagtail</a> stream field which allows composing content out of reusable blocks of content (<a href="https://406.ch/writing/i-just-learned-about-wagtail-s-streamfield/">sounds familiar!</a>). The result is saved in a JSON blob in the database with all the advantages and disadvantages that entails.</p> <p>Now, django-content-editor is a worthy competitor when you do not want to add another interface to your website besides the user-facing frontend and the Django administration interface.</p> <p>The example from the Lincoln Loop blog post can be replicated quite closely with django-content-editor by using sections. I&rsquo;m using the <a href="https://pypi.org/project/django-json-schema-editor/">django-json-schema-editor</a> package for the section plugin since it easily allows adding more fields if some section type needs it.</p> <p>Here&rsquo;s an example model definition:</p> <div class="chl"><pre><span></span><code><span class="c1"># Models</span> <span class="kn">from</span> <span class="nn">content_editor.models</span> <span class="kn">import</span> <span class="n">Region</span><span class="p">,</span> <span class="n">create_plugin_base</span> <span class="kn">from</span> <span class="nn">django_json_schema_editor.plugins</span> <span class="kn">import</span> <span class="n">JSONPluginBase</span> <span class="kn">from</span> <span class="nn">feincms3</span> <span class="kn">import</span> <span class="n">plugins</span> <span class="k">class</span> <span class="nc">Page</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">Model</span><span class="p">):</span> <span class="c1"># You have to define regions; each region gets a tab in the admin interface</span> <span class="n">regions</span> <span class="o">=</span> <span class="p">[</span><span class="n">Region</span><span class="p">(</span><span class="n">key</span><span class="o">=</span><span class="s2">&quot;content&quot;</span><span class="p">,</span> <span class="n">title</span><span class="o">=</span><span class="s2">&quot;Content&quot;</span><span class="p">)]</span> <span class="c1"># Additional fields for the page...</span> <span class="n">PagePlugin</span> <span class="o">=</span> <span class="n">create_plugin_base</span><span class="p">(</span><span class="n">Page</span><span class="p">)</span> <span class="k">class</span> <span class="nc">RichText</span><span class="p">(</span><span class="n">plugins</span><span class="o">.</span><span class="n">richtext</span><span class="o">.</span><span class="n">RichText</span><span class="p">,</span> <span class="n">PagePlugin</span><span class="p">):</span> <span class="k">pass</span> <span class="k">class</span> <span class="nc">Image</span><span class="p">(</span><span class="n">plugins</span><span class="o">.</span><span class="n">image</span><span class="o">.</span><span class="n">Image</span><span class="p">,</span> <span class="n">PagePlugin</span><span class="p">):</span> <span class="k">pass</span> <span class="k">class</span> <span class="nc">Section</span><span class="p">(</span><span class="n">JSONPluginBase</span><span class="p">,</span> <span class="n">PagePlugin</span><span class="p">):</span> <span class="k">pass</span> <span class="n">AccordionSection</span> <span class="o">=</span> <span class="n">Section</span><span class="o">.</span><span class="n">proxy</span><span class="p">(</span> <span class="s2">&quot;accordion&quot;</span><span class="p">,</span> <span class="n">schema</span><span class="o">=</span><span class="p">{</span><span class="s2">&quot;type&quot;</span><span class="p">:</span> <span class="s2">&quot;object&quot;</span><span class="p">,</span> <span class="p">{</span><span class="s2">&quot;properties&quot;</span><span class="p">:</span> <span class="p">{</span><span class="s2">&quot;title&quot;</span><span class="p">:</span> <span class="p">{</span><span class="s2">&quot;type&quot;</span><span class="p">:</span> <span class="s2">&quot;string&quot;</span><span class="p">}}}},</span> <span class="p">)</span> <span class="n">CloseSection</span> <span class="o">=</span> <span class="n">Section</span><span class="o">.</span><span class="n">proxy</span><span class="p">(</span> <span class="s2">&quot;close&quot;</span><span class="p">,</span> <span class="n">schema</span><span class="o">=</span><span class="p">{</span><span class="s2">&quot;type&quot;</span><span class="p">:</span> <span class="s2">&quot;object&quot;</span><span class="p">,</span> <span class="p">{</span><span class="s2">&quot;properties&quot;</span><span class="p">:</span> <span class="p">{}}},</span> <span class="p">)</span> </code></pre></div> <p>Here&rsquo;s the corresponding admin definition:</p> <div class="chl"><pre><span></span><code><span class="c1"># Admin</span> <span class="kn">from</span> <span class="nn">content_editor.admin</span> <span class="kn">import</span> <span class="n">ContentEditor</span> <span class="kn">from</span> <span class="nn">django_json_schema_editor.plugins</span> <span class="kn">import</span> <span class="n">JSONPluginInline</span> <span class="kn">from</span> <span class="nn">feincms3</span> <span class="kn">import</span> <span class="n">plugins</span> <span class="nd">@admin</span><span class="o">.</span><span class="n">register</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">Page</span><span class="p">)</span> <span class="k">class</span> <span class="nc">PageAdmin</span><span class="p">(</span><span class="n">ContentEditor</span><span class="p">):</span> <span class="n">inlines</span> <span class="o">=</span> <span class="p">[</span> <span class="n">plugins</span><span class="o">.</span><span class="n">richtext</span><span class="o">.</span><span class="n">RichTextInline</span><span class="o">.</span><span class="n">create</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">RichText</span><span class="p">),</span> <span class="n">plugins</span><span class="o">.</span><span class="n">image</span><span class="o">.</span><span class="n">ImageInline</span><span class="o">.</span><span class="n">create</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">Image</span><span class="p">),</span> <span class="n">JSONPluginInline</span><span class="o">.</span><span class="n">create</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">AccordionSection</span><span class="p">,</span> <span class="n">sections</span><span class="o">=</span><span class="mi">1</span><span class="p">),</span> <span class="n">JSONPluginInline</span><span class="o">.</span><span class="n">create</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">CloseSection</span><span class="p">,</span> <span class="n">sections</span><span class="o">=-</span><span class="mi">1</span><span class="p">),</span> <span class="p">]</span> </code></pre></div> <p>The somewhat cryptic <code>sections=</code> argument says how many levels of sections the individual blocks open or close.</p> <p>To render the content including accordions I&rsquo;d probably use a <a href="https://feincms3.readthedocs.io/en/latest/guides/rendering.html#using-marks">feincms3 renderer</a>. At the time of writing the renderer definition for sections is a bit tricky.</p> <div class="chl"><pre><span></span><code><span class="kn">from</span> <span class="nn">feincms3.renderer</span> <span class="kn">import</span> <span class="n">RegionRenderer</span><span class="p">,</span> <span class="n">render_in_context</span><span class="p">,</span> <span class="n">template_renderer</span> <span class="k">class</span> <span class="nc">PageRenderer</span><span class="p">(</span><span class="n">RegionRenderer</span><span class="p">):</span> <span class="k">def</span> <span class="nf">handle</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">plugins</span><span class="p">,</span> <span class="n">context</span><span class="p">):</span> <span class="n">plugins</span> <span class="o">=</span> <span class="n">deque</span><span class="p">(</span><span class="n">plugins</span><span class="p">)</span> <span class="k">yield from</span> <span class="bp">self</span><span class="o">.</span><span class="n">_handle</span><span class="p">(</span><span class="n">plugins</span><span class="p">,</span> <span class="n">context</span><span class="p">)</span> <span class="k">def</span> <span class="nf">_handle</span><span class="p">(</span><span class="bp">self</span><span class="p">,</span> <span class="n">plugins</span><span class="p">,</span> <span class="n">context</span><span class="p">,</span> <span class="o">*</span><span class="p">,</span> <span class="n">in_section</span><span class="o">=</span><span class="kc">False</span><span class="p">):</span> <span class="k">while</span> <span class="n">plugins</span><span class="p">:</span> <span class="k">if</span> <span class="nb">isinstance</span><span class="p">(</span><span class="n">plugins</span><span class="p">[</span><span class="mi">0</span><span class="p">],</span> <span class="n">models</span><span class="o">.</span><span class="n">Section</span><span class="p">):</span> <span class="n">section</span> <span class="o">=</span> <span class="n">plugins</span><span class="o">.</span><span class="n">popleft</span><span class="p">()</span> <span class="k">if</span> <span class="n">section</span><span class="o">.</span><span class="n">type</span> <span class="o">==</span> <span class="s2">&quot;close&quot;</span><span class="p">:</span> <span class="k">if</span> <span class="n">in_section</span><span class="p">:</span> <span class="k">return</span> <span class="c1"># Ignore close section plugins when not inside section</span> <span class="k">continue</span> <span class="k">if</span> <span class="n">section</span><span class="o">.</span><span class="n">type</span> <span class="o">==</span> <span class="s2">&quot;accordion&quot;</span><span class="p">:</span> <span class="k">yield</span> <span class="n">render_in_context</span><span class="p">(</span><span class="s2">&quot;accordion.html&quot;</span><span class="p">,</span> <span class="p">{</span> <span class="s2">&quot;title&quot;</span><span class="p">:</span> <span class="n">accordion</span><span class="o">.</span><span class="n">data</span><span class="p">[</span><span class="s2">&quot;title&quot;</span><span class="p">],</span> <span class="s2">&quot;content&quot;</span><span class="p">:</span> <span class="bp">self</span><span class="o">.</span><span class="n">_handle</span><span class="p">(</span><span class="n">plugins</span><span class="p">,</span> <span class="n">context</span><span class="p">,</span> <span class="n">in_section</span><span class="o">=</span><span class="kc">True</span><span class="p">),</span> <span class="p">})</span> <span class="k">else</span><span class="p">:</span> <span class="k">yield</span> <span class="bp">self</span><span class="o">.</span><span class="n">render_plugin</span><span class="p">(</span><span class="n">plugin</span><span class="p">,</span> <span class="n">context</span><span class="p">)</span> <span class="n">renderer</span> <span class="o">=</span> <span class="n">PageRenderer</span><span class="p">()</span> <span class="n">renderer</span><span class="o">.</span><span class="n">register</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">RichText</span><span class="p">,</span> <span class="n">template_renderer</span><span class="p">(</span><span class="s2">&quot;plugins/richtext.html&quot;</span><span class="p">))</span> <span class="n">renderer</span><span class="o">.</span><span class="n">register</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">Image</span><span class="p">,</span> <span class="n">template_renderer</span><span class="p">(</span><span class="s2">&quot;plugins/image.html&quot;</span><span class="p">))</span> <span class="n">renderer</span><span class="o">.</span><span class="n">register</span><span class="p">(</span><span class="n">models</span><span class="o">.</span><span class="n">Section</span><span class="p">,</span> <span class="s2">&quot;&quot;</span><span class="p">)</span> </code></pre></div> <h2 id="closing-thoughts"><a class="toclink" href="#closing-thoughts">Closing thoughts</a></h2> <p>Sometimes, I think to myself, I&rsquo;ll &ldquo;just&rdquo; write a &ldquo;simple&rdquo; blog post. I get what I deserve when using those forbidden words. This blog post is neither short or simple. That being said, the rendering code is a bit tricky, the rest is quite straightforward. The amount of code in django-content-editor and feincms3 is reasonable as well. Even though it may look like a lot you&rsquo;ll still be <a href="https://406.ch/writing/run-less-code-in-production-or-youll-end-up-paying-the-price-later/">running less code in production</a> than when using comparable solutions built using Django.</p>Weeknotes (2024 week 37)https://406.ch/writing/weeknotes-2024-week-37/2024-09-11T12:00:00Z2024-09-11T12:00:00Z<h1 id="weeknotes-2024-week-37"><a class="toclink" href="#weeknotes-2024-week-37">Weeknotes (2024 week 37)</a></h1> <h2 id="django-debug-toolbar-alpha-with-async-support"><a class="toclink" href="#django-debug-toolbar-alpha-with-async-support">django-debug-toolbar alpha with async support!</a></h2> <p>I have helped mentoring Aman Pandey who has worked all summer to add async support to <a href="https://github.com/jazzband/django-debug-toolbar/">django-debug-toolbar</a>. <a href="https://github.com/tim-schilling">Tim</a> has released an alpha which contains all of the work up to a few days ago. Test it! Let&rsquo;s find the breakages before the final release.</p> <h2 id="dropping-python-39-from-my-projects"><a class="toclink" href="#dropping-python-39-from-my-projects">Dropping Python 3.9 from my projects</a></h2> <p>I have read <a href="https://noumenal.es/posts/the-only-green-python/yLw/">Carlton&rsquo;s post about the only green Python release</a> and have started dropping Python 3.9 support from many of the packages I maintain. This is such a good point:</p> <blockquote> <p>[&hellip;] I’m also thinking about it in terms of reducing the number of Python versions we support in CI. It feels like a lot of trees to support 5 full versions of Python for their entire life. 🌳</p> </blockquote> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/django-debug-toolbar/5.0.0a0/">django-debug-toolbar 5.0.0a0</a>: See above.</li> <li><a href="https://pypi.org/project/form-designer/">form-designer 0.26.2</a>: The values of choice fields are now returned as-is when sending mails or exporting form submissions instead of only returning the slugified version.</li> <li><a href="https://pypi.org/project/django-authlib/">django-authlib 0.17.1</a>: The <a href="https://406.ch/writing/keep-content-managers-admin-access-up-to-date-with-role-based-permissions/">role-based permissions backend</a> had a bug where it wouldn&rsquo;t return all available permissions in all circumstances, leading to empty navigation sidebars in the Django administration. This has been fixed.</li> <li><a href="https://pypi.org/project/feincms3/">feincms3 5.2.3</a>: Bugfix release, the page moving interface is no longer hidden by an expanded navigation sidebar. I almost always turn off the sidebar in my projects so I haven&rsquo;t noticed this.</li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.8.1</a>: Contains the most recent ProseMirror updates and bugfixes.</li> <li><a href="https://pypi.org/project/django-content-editor/">django-content-editor 7.0.10</a>: Now supports sections. Separate blog post coming up!</li> </ul>Weeknotes (2024 week 35)https://406.ch/writing/weeknotes-2024-week-35/2024-08-28T12:00:00Z2024-08-28T12:00:00Z<h1 id="weeknotes-2024-week-35"><a class="toclink" href="#weeknotes-2024-week-35">Weeknotes (2024 week 35)</a></h1> <h2 id="getting-deep-into-htmx-and-django-template-partials"><a class="toclink" href="#getting-deep-into-htmx-and-django-template-partials">Getting deep into htmx and django-template-partials</a></h2> <p>I have been skeptical about <a href="https://htmx.org/">htmx</a> for some time because basically everything the library does is straightforward to do myself with a few lines of JavaScript. I am a convert now because, really, adding a few HTML attributes is nicer than copy pasting a few lines of JavaScript. Feels good.</p> <p>The combination of htmx with <a href="https://github.com/carltongibson/django-template-partials/">django-template-partials</a> is great as well. I didn&rsquo;t know I had been missing template partials until I started using them. Includes are still useful, but replacing some of them with partials makes working on the project much more enjoyable.</p> <p>I haven&rsquo;t yet had a use for <a href="https://django-htmx.readthedocs.io/">django-htmx</a> but I may yet surprise myself.</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/django-authlib/">django-authlib 0.17</a>: django-authlib bundles <code>authlib.little_auth</code> which offers an user model which uses the email address as the username. I have also introduced the concept of <a href="https://406.ch/writing/keep-content-managers-admin-access-up-to-date-with-role-based-permissions/">roles instead of permissions</a>; now I have reorganized the user admin fieldset to hide user permissions altogether. Group permissions are still available as are roles. I&rsquo;m personally convinced that user permissions were a mistake.</li> <li><a href="https://pypi.org/project/feincms3-forms/">feincms3-forms 0.5</a>: Allowed setting a maximum length for the bundled URL and email fields through the Django administration interface.</li> <li><a href="https://pypi.org/project/django-content-editor/">django-content-editor 7.0.7</a>: Fixed a bug where plugins with several fieldsets weren&rsquo;t collapsed completely.</li> <li><a href="https://pypi.org/project/django-imagefield/">django-imagefield 0.19.1</a>: Allowed deactivating autogeneration of thumbnails completely through the <code>IMAGEFIELD_AUTOGENERATE</code> setting. This is very useful for batch processing. Also, documented all available settings.</li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.8</a>: Added support for translating the interface elements and for restricting the available heading levels in the UI.</li> <li><a href="https://pypi.org/project/form-designer/">form-designer 0.25</a>: Fixed the type of the author field for the send-to-author processing action.</li> <li><a href="https://pypi.org/project/feincms3/">feincms3 5.2.2</a>: Added support for embedding <a href="https://www.srf.ch/play/tv">SRF play</a> external content. They do not support oEmbed unfortunately.</li> <li><a href="https://pypi.org/project/feincms3-cookiecontrol/">feincms3-cookiecontrol 1.5.3</a>: Added support for SRF play embeddings as well. The difference is that the feincms3-cookiecontrol embedding requires consent before embedding external content.</li> </ul>How I handle versioninghttps://406.ch/writing/how-i-handle-versioning/2024-08-20T12:00:00Z2024-08-20T12:00:00Z<h1 id="how-i-handle-versioning"><a class="toclink" href="#how-i-handle-versioning">How I handle versioning</a></h1> <p>I have been reading up on versioning methods a bit, and I noticed that I never shared my a bit unorthodox versioning method. I previously wrote about my <a href="https://406.ch/writing/my-rules-for-releasing-open-source-software/">rules for releasing open source software</a> but skipped everything related to versioning.</p> <p>I use something close to the ideas of <a href="https://semver.org/">semantic versioning</a> but it&rsquo;s not quite that. Note that the versioning scheme has nothing to do with production readiness, it&rsquo;s more about communicating the state of the project.</p> <h2 id="00x-everything-breaks"><a class="toclink" href="#00x-everything-breaks">0.0.x &ndash; Everything breaks</a></h2> <p>I don&rsquo;t trust my choices a lot. Everything may radically change, and I may also abandon the project with no hard feelings.</p> <h2 id="010-changelogging"><a class="toclink" href="#010-changelogging">0.1.0 &ndash; Changelogging</a></h2> <p>I generally start writing release notes or rather a <a href="https://en.wikipedia.org/wiki/Changelog">Changelog</a>, since writing full release notes is much more work. You absolutely have to test the software and I do not guarantee anything related to backwards compatibility &ndash; just that you&rsquo;ll know about breaking changes when you read the CHANGELOG.</p> <h2 id="x-bugfixes-and-pure-additions"><a class="toclink" href="#x-bugfixes-and-pure-additions">?.?.x &ndash; Bugfixes and pure additions</a></h2> <p>Strictly speaking, patch version increments are only for bugfixes. However, when I add new features which are purely additional to the package or if there&rsquo;s no way (famous last words) that anything will break, I&rsquo;ll upload a patch release.</p> <p>Sometimes it might be better to increment the minor version, yes. But, being very lazy, I sometimes only want to add a line to the CHANGELOG instead of adding a new heading for the new release. Also, if I don&rsquo;t quickly release new additions there&rsquo;s a real danger that I&rsquo;ll only come back to the project weeks or months later, and I don&rsquo;t want anyone (myself included) waiting on these updates. Also, if I inadvertently introduced new bugs it&rsquo;s better to fix them quickly instead of forgetting about it first and having to rediscover the buggy code later.</p> <h2 id="100-im-happy"><a class="toclink" href="#100-im-happy">1.0.0 &ndash; I&rsquo;m happy</a></h2> <p>I&rsquo;m happy with the package and I commit to start using semver more properly. If I have to (or want to) break backwards compatibility you get a migration path, especially if I need it myself. Which is often the case since I&rsquo;m maintaining dozens of Django-based websites and webapps, and I&rsquo;m always <a href="https://en.wikipedia.org/wiki/Eating_your_own_dog_food">dogfooding</a> my stuff.</p> <p>The pure additions case from the last section still applies, since slapping a 1.0 version on something doesn&rsquo;t suddenly make me less lazy.</p> <h2 id="outliers"><a class="toclink" href="#outliers">Outliers</a></h2> <p><a href="https://github.com/feincms/feincms">FeinCMS</a> uses a variant of <a href="https://calver.org/">calendar versioning</a>; the version at the time of writing is 24.8.2 which means the second release in August 2024.</p> <p>There are probably other outliers I forgot about.</p>Weeknotes (2024 week 33)https://406.ch/writing/weeknotes-2024-week-33/2024-08-14T12:00:00Z2024-08-14T12:00:00Z<h1 id="weeknotes-2024-week-33"><a class="toclink" href="#weeknotes-2024-week-33">Weeknotes (2024 week 33)</a></h1> <h2 id="partying"><a class="toclink" href="#partying">Partying</a></h2> <p>It&rsquo;s summer, it&rsquo;s hot, and it&rsquo;s dance week. <a href="https://lethargy.ch/">Lethargy</a> is over, <a href="https://www.junglestreetgroove.ch/">Jungle Street Groove</a> is coming up. Good times.</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/django-json-schema-editor/">django-json-schema-editor 0.1</a>: I have finally left the alpha versioning. I&rsquo;m still not committing to backwards compatibility, but I have started writing a CHANGELOG.</li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.7.1</a>: Thanks to Carlton&rsquo;s pull request I have finally cleaned up the CSS somewhat and made overriding the styles more agreeable when using the editor outside the Django administration. The confusing active state of menubar buttons has also been rectified. <a href="https://django-prose-editor.readthedocs.io/">Docs are now available on Read the Docs.</a></li> <li><a href="https://pypi.org/project/django-imagefield/">django-imagefield 0.19</a>: django-imagefield can now be used with proxy models. Previously, thumbnails weren&rsquo;t generated or deleted when saving proxy models because the signal handlers would only be called if the <code>sender</code> matches exactly. I have already debugged this before, but have forgotten about it again. The ticket is really old for this, and fixing it isn&rsquo;t easy since it&rsquo;s unclear what should happen (<a href="https://code.djangoproject.com/ticket/9318">#9318</a>).</li> <li><a href="https://pypi.org/project/django_canonical_domain/">django-canonical-domain 0.11</a>: django-canonical-domain has gained support for excluding additional domains from the canonical domain redirect. django-canonical-domain is used to redirect users to HTTPS (optionally) and to a particular canonical domain (as the name says). But sometimes you have auxiliary domains, e.g. for an API service, which shouldn&rsquo;t be redirected. The package can now be used in these scenarios as well.</li> <li><a href="https://pypi.org/project/FeinCMS/">FeinCMS 24.8.2</a>: The venerable FeinCMS, now more than 15 years old. The thumbnailing support had a bug where it tried saving JPEGs using RGBA (which obviously doesn&rsquo;t work). This has been fixed.</li> </ul>Weeknotes (2024 week 31)https://406.ch/writing/weeknotes-2024-week-31/2024-07-31T12:00:00Z2024-07-31T12:00:00Z<h1 id="weeknotes-2024-week-31"><a class="toclink" href="#weeknotes-2024-week-31">Weeknotes (2024 week 31)</a></h1> <p>I have missed almost two months of weeknotes. I&rsquo;ve got some catching up to do. I have tried writing a larger piece on my thoughts about CMS, but with everything going on in my personal and work life I haven&rsquo;t made much progress.</p> <p>This weeknotes entry is me trying to get back into the groove of writing (and publishing!) regularly.</p> <h2 id="django-prose-editor"><a class="toclink" href="#django-prose-editor"><a href="https://github.com/matthiask/django-prose-editor/">django-prose-editor</a></a></h2> <p>I have previously written about the <a href="https://prosemirror.net/">ProseMirror</a>-based editor for Django websites <a href="https://406.ch/writing/django-prose-editor-prose-editing-component-for-the-django-admin/">here</a>. I have continued working on the project in the meantime. Apart from bugfixes the big new feature is the support for showing typographic characters. For now the editor supports showing non-breaking spaces and soft hyphens. The project seems to get a little more interest after the deprecation of django-ckeditor has become more well known and the project has even received a contribution by someone else. It&rsquo;s always a lovely moment when this happens.</p> <h2 id="django-json-schema-editor"><a class="toclink" href="#django-json-schema-editor"><a href="https://github.com/matthiask/django-json-schema-editor">django-json-schema-editor</a></a></h2> <p>Still alpha. Updated the vendorized JSON editor and fixed the integration into Django to not throw errors with the newer version.</p> <p>Foreign key fields now support describing the referenced value similar to the raw ID fields functionality. Added optional support for using <code>"format": "prose"</code> to use the django-prose-editor to edit individual fields. JSON plugins for the content editor are now downcasted into their proxy models automatically. This is especially useful with the feincms3 changes mentioned below. (You do not have to use either django-content-editor or feincms3 to use this package!)</p> <p>The following screenshot shows the prose editor integration; the mentioned foreign key field description isn&rsquo;t visible yet here.</p> <p><img alt="A screenshot of the JSON editor including the prose editor" src="https://406.ch/assets/20240731-json-editor.png" /></p> <h2 id="traduire"><a class="toclink" href="#traduire"><a href="https://github.com/matthiask/traduire">Traduire</a></a></h2> <p>Traduire (french for «translate») is a web-based platform for editing gettext translations.</p> <p>It is intended as a replacement for Transifex, Weblate and comparable products. It is geared towards small teams or agencies which want to allow their customers and their less technical team members to update translations.</p> <p>Traduire profits from the great work done on <a href="https://github.com/mbi/django-rosetta/">django-rosetta</a>. I would still be using Rosetta if it would work when used with a container orchestator such as Kubernetes. Since all application storage is ephemeral that doesn&rsquo;t work, translation editing and deployment have to be separated.</p> <p><img alt="A screenshot of the Traduire interface" src="https://406.ch/assets/20240731-traduire.png" /></p> <p>It is built using Django and relies on <a href="https://pypi.org/project/polib/">polib</a> to do the heavy lifting.</p> <p>This is a project which might also be interesting for others. I would especially appreciate it if someone could contribute an easier way to get it up and running, e.g. using a Docker Compose configuration or something. I am using Kubernetes and GitOps to host it, but that&rsquo;s not straightforward at all. Really, all that&rsquo;s needed to run it is a Django host with any database which is supported by Django. I prefer using PostgreSQL because I have it, but sqlite etc. work just as well.</p> <h2 id="releases-since-the-second-week-of-june"><a class="toclink" href="#releases-since-the-second-week-of-june">Releases since the second week of June</a></h2> <ul> <li><a href="https://pypi.org/project/django-translated-fields/">django-translated-fields 0.13</a>: Nothing much except for CI and pre-commit updates. The implementation continues to be rock-solid and basically unchanged.</li> <li><a href="https://pypi.org/project/django-content-editor/">django-content-editor 7.0.6</a>: Tweaks and fixes to the new interface. Added better scrolling behavior when dragging content around. The editor now also supports colorized icons which helps quickly understanding the structure of some content when there are many plugins.</li> <li><a href="https://pypi.org/project/blacknoise/">blacknoise 1.0.2</a>: Fixed a few bugs in the <code>blacknoise.compress</code> utility and started running the testsuite on GitHub actions. <a href="https://github.com/evansd/whitenoise/">whitenoise</a> has been friendly-forked as <a href="https://github.com/Archmonger/ServeStatic">ServeStatic</a> and I&rsquo;m definitely having a close look at this project as well, but blacknoise is simple and works well, so I&rsquo;m not convinced that switching back to the much larger project (in terms of amounts of code) is an improvement now.</li> <li><a href="https://pypi.org/project/django-authlib/">django-authlib</a>: Minor bugfixes.</li> <li><a href="https://pypi.org/project/feincms3/">feincms3</a>: Allowed registering plugin models with the renderer which aren&rsquo;t supposed to be fetched from the database. This is especially useful when used together with JSON plugins, where the individual JSON plugins are created as proxies for the underlying Django model and fetched all at once. Disabled the version check on our CKEditor plugin. Still, really stop using CKEditor 4 if you want to use maintained software.</li> <li><a href="https://pypi.org/project/FeinCMS/">FeinCMS 24.7.1</a>: Small bugfixes, and made the Read the Docs build work correctly.</li> <li><a href="https://pypi.org/project/django-debug-toolbar/">django-debug-toolbar 4.4.x</a>: The toolbar continues to be a nice project to work on. Fixed a few edge cases in the new alerts panel.</li> <li><a href="https://pypi.org/project/django-admin-ordering/">django-admin-ordering 0.18.2</a>: The value of <code>ordering_field</code> now has additional sanity checks.</li> <li><a href="https://pypi.org/project/django-cabinet/">django-cabinet 0.16</a>: cabinet now supports exporting a folder as a ZIP file while preserving the structure you see in the CMS instead of the structure on the file system. The inline upload form has been dropped from the <code>CabinetForeignKey</code> widget because the folder dropdown slowed down the page a lot when used on a site with many folders. Using the raw ID fields popup isn&rsquo;t that bad.</li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.6.2</a>: See above.</li> <li><a href="https://pypi.org/project/django-json-schema-editor/">django-json-schema-editor 0.0.28</a>: See above.</li> </ul>Weeknotes (2024 week 23)https://406.ch/writing/weeknotes-2024-week-23/2024-06-07T12:00:00Z2024-06-07T12:00:00Z<h1 id="weeknotes-2024-week-23"><a class="toclink" href="#weeknotes-2024-week-23">Weeknotes (2024 week 23)</a></h1> <h2 id="switching-everything-from-pip-to-uv"><a class="toclink" href="#switching-everything-from-pip-to-uv">Switching everything from pip to uv</a></h2> <p>Enough said. I&rsquo;m always astonished how fast computers can be.</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/django-admin-ordering/">django-admin-ordering 0.18</a>: Added a database index to the ordering field since we&rsquo;re always sorting by it.</li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.4</a>: Dropped the jQuery dependency making it possible to use the editor outside the Django administration interface without annoying JavaScript errors. Allowed additional heading levels and moved the block type buttons into a popover.</li> <li><a href="https://pypi.org/project/django-debug-toolbar/">django-debug-toolbar 4.4.2</a>: I enjoy working on this important piece of software very much.</li> <li><a href="https://pypi.org/project/django-email-hosts/">django-email-hosts 0.2.1</a>: Added a command analogous to <code>./manage.py sendtestemail</code> so that it&rsquo;s possible to easily test the different configured email backends.</li> <li><a href="https://pypi.org/project/feincms3/">feincms3 5.0</a>: I completely reworked the move node action; previously it opened a new page where you could see all possible targets; now you can cut a page and paste it somewhere else. The advantages of the new interface is that you don&rsquo;t leave the changelist and can still profit from all its features while moving pages around.</li> <li><a href="https://pypi.org/project/feincms3-sites/">feincms3-sites 0.21</a>: A new release taking advantage of a new hook in feincms3 7.0 so that the new moving interface works.</li> <li><a href="https://pypi.org/project/django-authlib/">django-authlib 0.16.5</a>: authlib now shows a welcome message when authenticating using admin OAuth2. It&rsquo;s nice and helps with debugging strange authentication failures.</li> <li><a href="https://pypi.org/project/django-content-editor/">django-content-editor 7.0</a>: I reworked the UI. The sidebar is gone, instead there are nice buttons in the place where you can add new plugins; the plugins appear in a nice grid instead of a list, which looks much better once you have more than just a few plugin types available. Also, plugin type icons are now shown in the plugin forms. I think it looks much better than before.</li> <li><a href="https://pypi.org/project/feincms3-cookiecontrol/">feincms3-cookiecontrol 1.5.2</a>: I didn&rsquo;t contribute anything to this release which is also a nice experience for a change. The Google consent mode integration has been improved and simplified.</li> <li><a href="https://pypi.org/project/django-json-schema-editor/">django-json-schema-editor 0.0.22</a>: Various small-ish improvements. I should really start using higher version numbers, but not having to commit to anything also feels great. That being said, the editor is in active use in several projects, so maybe I&rsquo;m deceiving myself.</li> </ul>Workbench: Coffee time!https://406.ch/writing/workbench-coffee-time/2024-05-24T12:00:00Z2024-05-24T12:00:00Z<h1 id="workbench-coffee-time"><a class="toclink" href="#workbench-coffee-time">Workbench: Coffee time!</a></h1> <p>I have written about <a href="https://406.ch/writing/workbench-the-django-based-agency-software/">the Workbench agency software</a> a few weeks back.</p> <p>Back when we were using Slack at <a href="https://feinheit.ch/">Feinheit</a> we used a Donut bot to generate randomized invites for a coffee break. <a href="https://406.ch/writing/why-we-switched-from-slack-to-discord-at-work/">We lost that bot when we switched to Discord</a>. I searched some time for an equivalent bot for Discord but couldn&rsquo;t find any, so like any self-respecting nerd, break lover and NIH-sufferer I reimplemented the functionality as a part of our agency software.</p> <p>The first version generated totally random pairings without any history; I thought that maybe this would be good enough, but of course I received an invite for a coffee with the exact same person in the first two invitations. Even though I did enjoy the break both times this motivated me to put some effort into a better solution :-)</p> <p>The result of this work is the <a href="https://github.com/matthiask/workbench/blob/a14d8b9560def7a4e2bbf7531eb6108f734568db/workbench/accounts/tasks.py#L57"><code>coffee_invites()</code> function</a></p> <p>There are two main parts to the generator:</p> <ul> <li>It randomly generates twenty pairings from all participants; it prefers groups of two except for the last group which may contain three people.</li> <li>It loads old pairings from the database and gives higher penalties to equal pairings depending on the time which has passed since the last time.</li> </ul> <p>So, it&rsquo;s not mathematically perfect, but all imbalances will cancel out over a long time. I experimented with the discounting factor and the number of random pairings it generates upfront. The values in the code seem to work fine.</p> <p>Sometimes it&rsquo;s simple stuff like this which gets me going and reignites the love for programming. Like a small puzzle.</p>Weeknotes (2024 week 21)https://406.ch/writing/weeknotes-2024-week-21/2024-05-22T12:00:00Z2024-05-22T12:00:00Z<h1 id="weeknotes-2024-week-21"><a class="toclink" href="#weeknotes-2024-week-21">Weeknotes (2024 week 21)</a></h1> <p>There have been times when work has been more enjoyable than in the last few weeks. It feels more stressful than at other times, and this mostly has to do with particular projects. I hope I&rsquo;ll be able to move on soon.</p> <h2 id="blacknoise"><a class="toclink" href="#blacknoise">blacknoise</a></h2> <p>I have released <a href="https://pypi.org/project/blacknoise/">blacknoise 1.0</a>. It&rsquo;s an ASGI app for static file serving inspired by <a href="https://github.com/evansd/whitenoise/">whitenoise</a>.</p> <p>The 1.0 version number is only a big step in versioning terms, not much has happened with the code. It&rsquo;s a tiny little well working piece of software which has been running in production for some time without any hickups. The biggest recent change is that I have parallelized the gzip and brotli compression step; this makes building images using whitenoise more painful because there the wait is really really long sometimes. <a href="https://github.com/evansd/whitenoise/pull/484">A pull request fixing this exists</a>, but it hasn&rsquo;t moved forwards in months.</p> <p>I have written a longer post about it earlier this year <a href="https://406.ch/writing/blacknoise-asgi-app-for-static-file-serving/">here</a>.</p> <h2 id="releases"><a class="toclink" href="#releases">Releases</a></h2> <ul> <li><a href="https://pypi.org/project/feincms3-cookiecontrol/">feincms3-cookiecontrol 1.5</a>: Code golfing. Added backwards compatibility with old Django versions so that I can use it for old projects. Also includes optional support for Google consent management.</li> <li><a href="https://pypi.org/project/django_fast_export/">django-fast-export 0.1.1</a>: This is basically a repackaging of the streaming CSV view from Django&rsquo;s documentation as a reusable class. I have switched to using an iterator so that I can export even larger datasets.</li> <li><a href="https://pypi.org/project/django-json-schema-editor/">django-json-schema-editor 0.0.18</a>: Still alpha versioned but used in production in various projects. I should really release an 1.0 version, but there are no integration tests at all. Mainly visual tweaks in this update.</li> <li><a href="https://pypi.org/project/django-content-editor/">django-content-editor 6.5</a>: Better handling of templates and regions when a particular editor instance only shows a subset of configured templates. Disallowed adding plugins when in an unknown region. It&rsquo;s funny how many edge cases exist in software as old as this.</li> <li><a href="https://pypi.org/project/blacknoise/">blacknoise 1.0</a>: See above.</li> <li><a href="https://pypi.org/project/html-sanitizer/">html-sanitizer 2.4.4</a>: Fixed edge cases with whitespace handling when merging elements.</li> <li><a href="https://pypi.org/project/feincms3-data/">feincms3-data 0.6.1</a>: Allowed <code>./manage.py f3loaddata -</code> to load JSON data from stdin.</li> </ul>Weeknotes (2024 week 18)https://406.ch/writing/weeknotes-2024-week-18/2024-05-03T12:00:00Z2024-05-03T12:00:00Z<h1 id="weeknotes-2024-week-18"><a class="toclink" href="#weeknotes-2024-week-18">Weeknotes (2024 week 18)</a></h1> <h2 id="google-summer-of-code-has-begun"><a class="toclink" href="#google-summer-of-code-has-begun">Google Summer of Code has begun</a></h2> <p>We have a student helping out with adding async support to the <a href="https://github.com/jazzband/django-debug-toolbar/">Django Debug Toolbar</a>. It&rsquo;s great that someone can spend some concentrated time to work on this. Tim and others have done all the necessary preparation work, I&rsquo;m only helping from the sidelines so don&rsquo;t thank me.</p> <h2 id="bike-to-work"><a class="toclink" href="#bike-to-work">Bike to Work</a></h2> <p>Two teams from my company are participating in the <a href="https://www.biketowork.ch/">Bike to Work Challenge 2024</a>. It&rsquo;s what I do anyway (if I&rsquo;m not working from home) but maybe it helps build others some motivation to get on the bicycle once more. Public transports in the city where I live are great but I&rsquo;ll always take the bike when I can. I also went on my first mountain bike ride in a few months yesterday, good fun.</p> <h2 id="json-blobs-and-referential-integrity"><a class="toclink" href="#json-blobs-and-referential-integrity">JSON blobs and referential integrity</a></h2> <p>The <a href="https://github.com/matthiask/django-json-schema-editor/">django-json-schema-editor</a> has gained support for referencing Django models. Here&rsquo;s an example schema excerpt:</p> <div class="chl"><pre><span></span><code><span class="p">{</span> <span class="w"> </span><span class="o">...</span> <span class="w"> </span><span class="s">&quot;articles&quot;</span><span class="p">:</span><span class="w"> </span><span class="p">{</span> <span class="w"> </span><span class="s">&quot;type&quot;</span><span class="p">:</span><span class="w"> </span><span class="s">&quot;array&quot;</span><span class="p">,</span> <span class="w"> </span><span class="s">&quot;format&quot;</span><span class="p">:</span><span class="w"> </span><span class="s">&quot;table&quot;</span><span class="p">,</span> <span class="w"> </span><span class="s">&quot;title&quot;</span><span class="p">:</span><span class="w"> </span><span class="nx">_</span><span class="p">(</span><span class="s">&quot;articles&quot;</span><span class="p">),</span> <span class="w"> </span><span class="s">&quot;minItems&quot;</span><span class="p">:</span><span class="w"> </span><span class="mi">1</span><span class="p">,</span> <span class="w"> </span><span class="s">&quot;maxItems&quot;</span><span class="p">:</span><span class="w"> </span><span class="mi">3</span><span class="p">,</span> <span class="w"> </span><span class="s">&quot;items&quot;</span><span class="p">:</span><span class="w"> </span><span class="p">{</span> <span class="w"> </span><span class="s">&quot;type&quot;</span><span class="p">:</span><span class="w"> </span><span class="s">&quot;string&quot;</span><span class="p">,</span> <span class="w"> </span><span class="s">&quot;title&quot;</span><span class="p">:</span><span class="w"> </span><span class="nx">_</span><span class="p">(</span><span class="s">&quot;article&quot;</span><span class="p">),</span> <span class="w"> </span><span class="s">&quot;format&quot;</span><span class="p">:</span><span class="w"> </span><span class="s">&quot;foreign_key&quot;</span><span class="p">,</span> <span class="w"> </span><span class="s">&quot;options&quot;</span><span class="p">:</span><span class="w"> </span><span class="p">{</span> <span class="w"> </span><span class="s">&quot;url&quot;</span><span class="p">:</span><span class="w"> </span><span class="s">&quot;/admin/articles/article/?_popup=1&amp;_to_field=id&quot;</span><span class="p">,</span> <span class="w"> </span><span class="p">},</span> <span class="w"> </span><span class="p">},</span> <span class="w"> </span><span class="p">},</span> <span class="w"> </span><span class="o">...</span> <span class="p">}</span> </code></pre></div> <p>The ID field is stringly typed; using an integer directly wouldn&rsquo;t work because the empty string isn&rsquo;t a valid integer.</p> <p>The problem with referencing models in this way is that there&rsquo;s no way to know if the referenced object is still around or not, or even to protect it against deletion. The bundled django-content-editor <code>JSONPlugin</code> now supports automatically generating a <code>ManyToManyField</code> with a <code>through</code> model which protects articles from deletion as long as they are referenced from a <code>JSONPlugin</code> instance. The <code>register_reference</code> line creates the mentioned model with an <code>on_delete=models.PROTECT</code> foreign key to articles and a <code>post_save</code> handler which updates said references.</p> <div class="chl"><pre><span></span><code><span class="kn">from</span><span class="w"> </span><span class="nn">django_json_schema_editor.plugins</span><span class="w"> </span><span class="kn">import</span><span class="w"> </span><span class="n">JSONPluginBase</span><span class="p">,</span><span class="w"> </span><span class="n">register_reference</span> <span class="kn">from</span><span class="w"> </span><span class="nn">articles.models</span><span class="w"> </span><span class="kn">import</span><span class="w"> </span><span class="n">Article</span> <span class="k">class</span><span class="w"> </span><span class="nc">JSONPlugin</span><span class="p">(</span><span class="n">JSONPluginBase</span><span class="p">,</span><span class="w"> </span><span class="o">...</span><span class="p">):</span> <span class="w"> </span><span class="k">pass</span> <span class="n">register_reference</span><span class="p">(</span><span class="n">JSONPlugin</span><span class="p">,</span><span class="w"> </span><span class="s2">&quot;articles&quot;</span><span class="p">,</span><span class="w"> </span><span class="n">Article</span><span class="p">)</span> </code></pre></div> <h2 id="releases-since-the-beginning-of-april"><a class="toclink" href="#releases-since-the-beginning-of-april">Releases since the beginning of April</a></h2> <ul> <li><a href="https://pypi.org/project/django-json-schema-editor/">django-json-schema-editor 0.0.14</a>: See above. Also, some styling work and a patch update to the vendorized json-editor.</li> <li><a href="https://pypi.org/project/django-content-editor/">django-content-editor 6.4.6</a>: Many small stylistic fixes. The target indicator when dragging plugins is now also shown when plugins are collapsed. It&rsquo;s now possible to directly drag a plugin to the end, and not just to the second to last position.</li> <li><a href="https://pypi.org/project/django-prose-editor/">django-prose-editor 0.3.4</a>: Switched to the nh3 sanitizer because it&rsquo;s faster and because ProseMirror never emits HTML which has to be cleaned up first. Stopped generating menu items for nodes and marks which aren&rsquo;t in the schema. Added the possibility to reduce the functionality per editor instance. Small tweaks and fixes.</li> <li><a href="https://pypi.org/project/django-tree-queries/">django-tree-queries 0.19</a>: Added support for pre-filtering the tree (much more efficient when only querying a part of the tree). Added support for adding additional fields to the CTE so that you can collect values from ancestors for other fields than the default fields too.</li> <li><a href="https://pypi.org/project/FeinCMS/">FeinCMS 24.4.2</a>: Added support for webp images. Fixed a few of the admin list filters to work with Django 5.</li> <li><a href="https://pypi.org/project/django-cabinet/">django-cabinet 0.14.3</a>: Fixed the support for the <code>extra_context</code> argument to our <code>changelist_view</code> implementation.</li> </ul>Workbench, the Django-based agency softwarehttps://406.ch/writing/workbench-the-django-based-agency-software/2024-04-24T12:00:00Z2024-04-24T12:00:00Z<h1 id="workbench-the-django-based-agency-software"><a class="toclink" href="#workbench-the-django-based-agency-software">Workbench, the Django-based agency software</a></h1> <p>I get the impression that there&rsquo;s a lot of interesting but unknown software in Django land. I don&rsquo;t know if there&rsquo;s any interest in some of the packages I have been working on; if not this blog post is for myself only.</p> <h2 id="history-time"><a class="toclink" href="#history-time">(Hi)story time</a></h2> <p>As people may know I work at <a href="https://feinheit.ch/">Feinheit</a>, an agency which specializes in digital communication services for SMEs, campaigns for referendums, and website and webapp development. At the time of writing we are a team of about 20-25 communication experts, graphic designers, programmers and project managers.</p> <p>We have many different clients and are working on many different projects at the same time and are billing by the hour<sup id="fnref:bythehour"><a class="footnote-ref" href="#fn:bythehour">1</a></sup>. Last year my own work has been billed to more than 50 different customers. In the early days we used a shared file server with spreadsheet files to track our working hours. Luckily we didn&rsquo;t often overwrite the edits others made but that was definitely something which happened from time to time.</p> <p>We knew of another agency who had the same problems and used a FileMaker-based software. Their solution had several problems, among them the fact that it became hard to evolve and that it got slower and slower as more and more data was entered into it over the years. They had the accounting know how and we had the software engineering know how so we wrote a webapp based on the Django framework. As always, it was much more work than the initial estimate, but if we as programmers didn&rsquo;t underestimate the effort needed we wouldn&rsquo;t have started many of the great projects we&rsquo;re now getting much value and/or enjoyment from, hopefully both. The product of that work was <a href="https://www.fineware.ch/">Metronom</a>. The first release happened a little bit later than <a href="https://www.getharvest.com/about">harvest</a> but it already came with full time tracking including an annual working time calculator, absence management, offers, invoices including PDF generation etc, so it was quite a bit more versatile while still being easier to use than &ldquo;real&rdquo; business software solutions.</p> <p>I personally was of the opinion that the product was good enough to try selling it, but for a variety of reasons (which I don&rsquo;t want to go into here) this never happened and <a href="https://feinheit.ch/">we</a> decided that we didn&rsquo;t want to be involved anymore.</p> <p>However, this meant that we were dead-end street with a software that didn&rsquo;t belong to us anymore, which wasn&rsquo;t evolving to our changing requirements. I also didn&rsquo;t enjoy working on it anymore. Over the years I have tried replacing it several times but that never came to pass until some time after the introduction of <a href="https://www.holacracy.org/">Holacracy</a> at our company. I noticed that I didn&rsquo;t have to persuade everyone but that I, as the responsible person for this particular decision, could &ldquo;just&rdquo;<sup id="fnref:just"><a class="footnote-ref" href="#fn:just">2</a></sup> move ahead with a broad interpretation of the purpose and accountabilities of one of my roles.</p> <h2 id="workbench"><a class="toclink" href="#workbench"><a href="https://github.com/matthiask/workbench">Workbench</a></a></h2> <p><img alt="Screenshot" src="https://406.ch/assets/20240424-workbench.png" /></p> <p><a href="https://github.com/matthiask/workbench">Workbench</a> is the product of a few long nights of hacking. The project was started as an experiment in 2015 and was used for sending invoices but that wasn&rsquo;t really the intended purpose. After long periods of lying dormant I have brought the project to a good enough state and switched Feinheit away from Metronom in 2019.</p> <p>I have thought long and hard about switching to one of the off-the-shelf products and it could very well be that one of them would work well for us. Also, we wouldn&rsquo;t have to pay (in form of working hours) for the maintenance and for enhancements ourselves. On the other hand, we can use a tool which is tailored to our needs. Is it worth the effort? That&rsquo;s always hard to answer. The tool certainly works well for the few companies which are using it right now, so there&rsquo;s no reason to agonize over that.</p> <p>At the time of writing, Workbench offers projects and services, offers and invoices incl. PDF generation, recurring invoices, an address book, a logbook for rendered services, annual working time reports, an acquisition funnel, a stupid project planning and resource management tool and various reports. It has not only replaced Metronom but also a selection of SaaS (for example Pipedrive and TeamGantt) we were using previously.</p> <p>The whole thing is open source because I don&rsquo;t want to try making agency software into a business anymore, I only want to solve the problems <em>we</em> have. That being said, if someone else finds it useful then that&rsquo;s certainly alright as well.</p> <p>The license has been MIT for the first few years but I have switched to the GPL because I wanted to integrate the excellent <a href="https://pypi.org/project/qrbill/">qrbill</a> module which is also licensed under the GPL. As I wrote elsewhere, I have released almost everything under the GPL in my first few open source years but have switched to BSD/MIT later when starting to work mainly with Python and Django because I thought that this license is a better fit<sup id="fnref:license"><a class="footnote-ref" href="#fn:license">3</a></sup> for the ecosystem. That being said, for a product such as this the GPL is certainly an excellent fit.</p> <h2 id="final-words"><a class="toclink" href="#final-words">Final words?</a></h2> <p>There are a few things I could write about to make this into a series. I&rsquo;m putting a few ideas here, not as an announcement, just as a reminder for myself.</p> <ul> <li>Better time tracking</li> <li>Patterns for creating recurring invoices</li> <li>Feature switches and configurability</li> <li><a href="https://406.ch/writing/workbench-coffee-time/">Coffee time</a></li> </ul> <div class="footnote"> <hr /> <ol> <li id="fn:bythehour"> <p>Rather, in six minute increments. It&rsquo;s even worse.&#160;<a class="footnote-backref" href="#fnref:bythehour" title="Jump back to footnote 1 in the text">&#8617;</a></p> </li> <li id="fn:just"> <p>Of course it&rsquo;s never that simple, because the responsibility is a lot. The important point is: It would have been a lot of work anyways, the big difference is that it was sufficient to get consent from people; no consensus required.&#160;<a class="footnote-backref" href="#fnref:just" title="Jump back to footnote 2 in the text">&#8617;</a></p> </li> <li id="fn:license"> <p>That&rsquo;s not meant as a criticism in any way!&#160;<a class="footnote-backref" href="#fnref:license" title="Jump back to footnote 3 in the text">&#8617;</a></p> </li> </ol> </div>