How to create apphooks¶
An apphook allows you to attach a Django application to a page. For example, you might have a news application that you’d like integrated with django CMS. In this case, you can create a normal django CMS page without any content of its own, and attach the news application to the page; the news application’s content will be delivered at the page’s URL.
All URLs under that page’s URL path will be passed to the attached application’s URLconf.
For the conceptual model (“mount points” and what it means for request routing), see Application hooks (“apphooks”).
The Tutorials section contains a basic guide to getting started with apphooks. This document assumes more familiarity with the CMS generally.
The basics of apphook creation¶
To create an apphook:
Create a
cms_apps.pyfile in your application.Add a
CMSAppsub-class.Register it with the apphook pool.
The file needs to contain a CMSApp sub-class. For
example:
from cms.app_base import CMSApp
from cms.apphook_pool import apphook_pool
@apphook_pool.register
class MyApphook(CMSApp):
app_name = "myapp" # must match the application namespace
name = "My Apphook"
def get_urls(self, page=None, language=None, **kwargs):
return ["myapp.urls"] # replace this with the path to your application's URLs module
Apphooks for namespaced applications¶
Your application should use namespaced URLs.
In the example above, the application uses the myapp namespace. Your CMSApp
sub-class must reflect the application’s namespace in the app_name attribute.
The application may specify a namespace by supplying an app_name in its urls.py,
or its documentation might advise that when you include its URLs, you do it like this:
path("myapp/", include("myapp.urls", app_name="myapp"))
If you fail to do this, then any templates in the application that invoke URLs using the
form {% url 'myapp:index' %} or views that call (for example)
reverse('myapp:index') will throw a NoReverseMatch error.
Apphooks for non-namespaced applications¶
If you are writing apphooks for third-party applications, you may find one that in fact does not have an application namespace for its URLs. Such an application is liable to run into namespace conflicts, and doesn’t represent good practice.
However if you do encounter such an application, your own apphook for it will need in
turn to forgo the app_name attribute.
Note that apphooks without app_name attributes can be attached only to one page at a
time; attempting to apply them a second time will cause an error. Only one instance of
these apphooks can exist.
See Attaching an application multiple times for more on having multiple apphook instances.
Returning apphook URLs manually¶
Instead of defining the URL patterns in another file myapp/urls.py, it also is
possible to return them manually, for example if you need to override the set provided.
An example:
from django.urls import path
from myapp.views import SomeListView, SomeDetailView
class MyApphook(CMSApp):
# ...
def get_urls(self, page=None, language=None, **kwargs):
return [
path("<str:slug>/", SomeDetailView.as_view()),
path("", SomeListView.as_view()),
]
However, it’s much neater to keep them in the application’s urls.py, where they can
easily be reused.
Loading new and re-configured apphooks¶
Certain apphook-related changes require server restarts in order to be loaded.
Whenever you:
add or remove an apphook
change the slug of a page containing an apphook or the slug of a page which has a descendant with an apphook
the URL caches must be reloaded.
If you have the cms.middleware.utils.ApphookReloadMiddleware installed, which is recommended, the server will do it for you by re-initialising the URL patterns automatically.
Otherwise, you will need to restart the server manually.
Using an apphook¶
Once your apphook has been set up and loaded, you’ll now be able to select the Application that’s hooked into that page from its Advanced settings.
Note
An apphook won’t actually do anything until the page it belongs to is published. Take note that this also means all parent pages must also be published.
The apphook attaches all of the apphooked application’s URLs to the page; its root URL will be the page’s own URL, and any lower-level URLs will be on the same URL path.
So, given an application with the urls.py for the views index_view and
archive_view:
urlpatterns = [
path("archive/", archive_view),
path("", index_view),
]
attached to a page whose URL path is /hello/world/, the views will be exposed as
follows:
index_viewat/hello/world/archive_viewat/hello/world/archive/
Using placeholders on an apphooked page¶
Added in version 5.1.
Even though the URL routing is delegated to your application, the page still exists and can
carry content in its placeholders. The placeholder template tag can therefore be used
in your apphook’s templates.
By default (when the apphook view does not call request.toolbar.set_object()), the
placeholder tag resolves against the current page’s
PageContent, exactly as it would on a regular CMS page —
including inherit walking up the page tree.
For example, in the template used by your app’s view:
{% load cms_tags %}
{% block content %}
{% placeholder "content" %}
{# application output below #}
{% endblock %}
This works because django CMS sets request.current_page for apphook requests.
Note
If your apphook view calls request.toolbar.set_object(some_model_instance) with
an instance of a model that uses
PlaceholderRelationField, the placeholder tag
will instead resolve against that object’s placeholders (and inherit has no
effect in this case). Set the toolbar object only when you want editors to work on
that custom model rather than on the page.
When a content editor opens the structure board on an apphooked page, django CMS
needs to know which template the apphook’s root view renders, should it be different
from the page’s template. This is so it can show the placeholders declared in that template.
The rendered template is provided by get_root_template().
By default, get_root_template inspects the URL pattern matching the empty
path inside get_urls() and reads its template_name.
This works automatically for class-based views in either of these forms:
# template_name set on the class
class HomeView(TemplateView):
template_name = "myapp/home.html"
urlpatterns = [path("", HomeView.as_view())]
# or passed as an as_view() keyword argument
urlpatterns = [path("", TemplateView.as_view(template_name="myapp/home.html"))]
Inference cannot work in some cases, in which the lookup falls back to None:
function-based views — the template name lives inside the view body, not on the callable
views that select their template at runtime
views that return a fully-rendered
HttpResponse(for example via therender()shortcut)
For these cases, override get_root_template() on the
apphook class. The override receives the same page and language arguments
as get_urls(), so the returned template can vary per
page or language:
class MyApphook(CMSApp):
name = _("My Apphook")
def get_urls(self, page=None, language=None, **kwargs):
return ["myapp.urls"]
def get_root_template(self, page=None, language=None, **kwargs):
return "myapp/home.html"
Make sure that the template you return here matches the one your root view actually renders, otherwise the placeholders shown in the structure board will diverge from those rendered on the page.
Sub-pages of an apphooked page¶
Usually you should not add child pages to a page with an apphook. This is because the apphook “swallows” all URLs below that page, handing them over to the attached application.
In the rare occasion that you nevertheless want to add child pages below an apphooked page, then you must add a special URL pattern to route requests back into the CMS.
For example, if you have an apphooked page at /hello/ and you want to add a CMS page,
and optionally its children below that page using the slug world, then rewrite the
URL patterns from above as:
from django.urls import path, re_path
from cms.views import details
def reroute_cms_page(request, path, page=None):
language = get_language_from_request(request, check_path=True)
return details(request, f'{page.get_path(language)}/{path}')
class MyApphook(CMSApp):
# ...
def get_urls(self, page=None, language=None, **kwargs):
return [
path("archive/", archive_view),
re_path(r"^(?P<path>world/.*)$", reroute_cms_page, {"page": page}),
path("", index_view),
]
Here we created a short function-based view named reroute_cms_page. It handles
the requests which otherwise would be swallowed by the apphook.
A request starting with the URL /hello/ then is handled by index_view,
/hello/archive/ is handled by archive_view, and /hello/world/,
/hello/world/foo, etc. are handled by our special view reroute_cms_page,
routing the request back to the detail() view of Django-CMS.
Managing apphooks¶
Uninstalling an apphook with applied instances¶
If you remove an apphook class from your system (in effect uninstalling it) that still has instances applied to pages, django CMS tries to handle this as gracefully as possible:
Affected pages still maintain a record of the applied apphook; if the apphook class is subsequently reinstated, it will work as before.
The page list will show apphook indicators where appropriate.
The page will otherwise behave like a normal django CMS page, and display its placeholders in the usual way.
If you save the page’s Advanced settings, the apphook will be removed.
Management commands¶
You can clear uninstalled apphook instances using the CMS management command uninstall
apphooks. For example:
python -m manage cms uninstall apphooks MyApphook MyOtherApphook
You can get a list of installed apphooks using the cms list; in this case:
python -m manage cms list apphooks
See the Management commands reference for more information.
Managing permissions on apphooks¶
By default the content represented by an apphook has the same permissions set as the page it is assigned to. So if for example a page requires the user to be logged in, then the attached apphook and all its URLs will have the same requirements.
To disable this behaviour set permissions = False on your apphook:
class MyApphook(CMSApp):
[...]
permissions = False
If you still want some of your views to use the CMS’s permission checks you can enable
them via a decorator, cms.utils.decorators.cms_perms
Here is a simple example:
from cms.utils.decorators import cms_perms
@cms_perms
def my_view(request, **kw):
...
If you make your own permission checks in your application, then use the
exclude_permissions property of the apphook:
class MyApphook(CMSApp):
[...]
permissions = True
exclude_permissions = ["some_nested_app"]
where you provide the name of the application in question
Automatically restart server on apphook changes¶
As mentioned above, whenever you:
add or remove an apphook
change the slug of a page containing an apphook
change the slug of a page with a descendant with an apphook
The CMS server will reload its URL caches. It does this by listening for the signal
cms.signals.urls_need_reloading.
Warning
This signal does not actually do anything itself. For automated server restarting you need to implement logic in your project that gets executed whenever this signal is fired. Because there are many ways of deploying Django applications, there is no way we can provide a generic solution for this problem that will always work.
The signal is fired after a request - for example, upon saving a page’s settings. If you change an apphook’s setting via an API the signal will not fire until a subsequent request.