python / expert
Snippet
Django Custom Template Loader mit Plugin-Architektur
Dieses Snippet implementiert eine plugin-basierte Template-Loader-Architektur für Django. Der `PluginLoader` koordiniert mehrere Template-Plugins durch ein Registry, jedes mit spezifischen Namespaces. Plugins implementieren `can_handle`, `fetch_source` und `render_template` Methoden, was das Laden von Templates aus Datenbanken, APIs oder benutzerdefinierten Storage-Systemen ermöglicht.
snippet.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
import osimport hashlibfrom django.template import TemplateDoesNotExistfrom django.template.loaders.base import Loaderfrom myapp.template_plugins import registryclass PluginLoader(Loader):def __init__(self, engine):super().__init__(engine)self.template_plugins = registry.get_template_sources()def get_template_sources(self, template_name):for plugin in self.template_plugins:if plugin.can_handle(template_name):source = plugin.fetch_source(template_name)if source:contents, path = sourceyield self._make_origin(path, template_name, plugin.name)def _make_origin(self, path, template_name, source_name):return f"{source_name}:{template_name}"def load_template(self, template_name, dirs=None):for origin in self.get_template_sources(template_name):try:contents = origin.split(':')[1] if ':' in origin else Nonefor plugin in self.template_plugins:if plugin.name == origin.split(':')[0]:template_string = plugin.render_template(template_name)return self.compile_template(template_string), originexcept Exception as e:continueraise TemplateDoesNotExist(template_name)class TemplatePlugin:def __init__(self, name, priority=50):self.name = nameself.priority = prioritydef can_handle(self, template_name):return template_name.startswith(f"{self.name}:")def fetch_source(self, template_name):raise NotImplementedErrordef render_template(self, template_name):raise NotImplementedError
django
Erklärung
1
class PluginLoader(Loader)
Benutzerdefinierter Django Template Loader erweitert Basis Loader Klasse
2
self.template_plugins = registry.get_template_sources()
Sammelt alle registrierten Plugins sortiert nach Priorität bei Initialisierung
3
def can_handle(self, template_name)
Plugin entscheidet via Namespace-Check ob es Template verarbeiten kann
4
def fetch_source(self, template_name)
Plugin fetcht rohe Template-Quelle aus seinem Storage-Backend
5
raise TemplateDoesNotExist(template_name)
Finale Exception wenn kein Plugin das angeforderte Template liefern kann