In Django 3.1, django.conf.urls.url() is deprecated in favor of django.urls.re_path(). https://docs.djangoproject.com/en/4.0/releases/3.1/#id2 Change-Id: I484694f8718f61c022126a1935cf28fce075894b
18 KiB
Tutorial: Building a Dashboard using Horizon
This tutorial covers how to use the various components in horizon to build an example dashboard and a panel with a tab which has a table containing data from the back end.
As an example, we'll create a new My Dashboard
dashboard
with a My Panel
panel that has an
Instances Tab
tab. The tab has a table which contains the
data pulled by the Nova instances API.
Note
This tutorial assumes you have either a devstack
or
openstack
environment up and running. There are a variety
of other resources which may be helpful to read first. For example, you
may want to start with the quickstart
or the Django tutorial.
Creating a dashboard
The quick version
Horizon provides a custom management command to create a typical base dashboard structure for you. Run the following commands in your Horizon root directory. It generates most of the boilerplate code you need:
$ mkdir openstack_dashboard/dashboards/mydashboard
$ tox -e manage -- startdash mydashboard \
--target openstack_dashboard/dashboards/mydashboard
$ mkdir openstack_dashboard/dashboards/mydashboard/mypanel
$ tox -e manage -- startpanel mypanel \
--dashboard=openstack_dashboard.dashboards.mydashboard \
--target=openstack_dashboard/dashboards/mydashboard/mypanel
You will notice that the directory mydashboard
gets
automatically populated with the files related to a dashboard and the
mypanel
directory gets automatically populated with the
files related to a panel.
Structure
If you use the tree mydashboard
command to list the
mydashboard
directory in
openstack_dashboard/dashboards
, you will see a directory
structure that looks like the following:
mydashboard
├── dashboard.py
├── __init__.py
├── mypanel
│ ├── __init__.py
│ ├── panel.py
│ ├── templates
│ │ └── mypanel
│ │ └── index.html
│ ├── tests.py
│ ├── urls.py
│ └── views.py
├── static
│ └── mydashboard
│ ├── scss
│ │ └── mydashboard.scss
│ └── js
│ └── mydashboard.js
└── templates
└── mydashboard
└── base.html
For this tutorial, we will not deal with the static directory, or the
tests.py
file. Leave them as they are.
With the rest of the files and directories in place, we can move on to add our own dashboard.
Defining a dashboard
Open the dashboard.py
file. You will notice the
following code has been automatically generated:
from django.utils.translation import gettext_lazy as _
import horizon
class Mydashboard(horizon.Dashboard):
name = _("Mydashboard")
slug = "mydashboard"
panels = () # Add your panels here.
default_panel = '' # Specify the slug of the dashboard's default panel.
horizon.register(Mydashboard)
If you want the dashboard name to be something else, you can change
the name
attribute in the dashboard.py
file .
For example, you can change it to be My Dashboard
:
name = _("My Dashboard")
A dashboard class will usually contain a name
attribute
(the display name of the dashboard), a slug
attribute (the
internal name that could be referenced by other components), a list of
panels, default panel, etc. We will cover how to add a panel in the next
section.
Creating a panel
We'll create a panel and call it My Panel
.
Structure
As described above, the mypanel
directory under
openstack_dashboard/dashboards/mydashboard
should look like
the following:
mypanel
├── __init__.py
├── panel.py
├── templates
│ └── mypanel
│ └── index.html
├── tests.py
├── urls.py
└── views.py
Defining a panel
The panel.py
file referenced above has a special
meaning. Within a dashboard, any module name listed in the
panels
attribute on the dashboard class will be
auto-discovered by looking for the panel.py
file in a
corresponding directory (the details are a bit magical, but have been
thoroughly vetted in Django's admin codebase).
Open the panel.py
file, you will have the following
auto-generated code:
from django.utils.translation import gettext_lazy as _
import horizon
from openstack_dashboard.dashboards.mydashboard import dashboard
class Mypanel(horizon.Panel):
name = _("Mypanel")
slug = "mypanel"
dashboard.Mydashboard.register(Mypanel)
If you want the panel name to be something else, you can change the
name
attribute in the panel.py
file . For
example, you can change it to be My Panel
:
name = _("My Panel")
Open the dashboard.py
file again, insert the following
code above the Mydashboard
class. This code defines the
Mygroup
class and adds a panel called
mypanel
:
class Mygroup(horizon.PanelGroup):
slug = "mygroup"
name = _("My Group")
panels = ('mypanel',)
Modify the Mydashboard
class to include
Mygroup
and add mypanel
as the default
panel:
class Mydashboard(horizon.Dashboard):
name = _("My Dashboard")
slug = "mydashboard"
panels = (Mygroup,) # Add your panels here.
default_panel = 'mypanel' # Specify the slug of the default panel.
The completed dashboard.py
file should look like the
following:
from django.utils.translation import gettext_lazy as _
import horizon
class Mygroup(horizon.PanelGroup):
slug = "mygroup"
name = _("My Group")
panels = ('mypanel',)
class Mydashboard(horizon.Dashboard):
name = _("My Dashboard")
slug = "mydashboard"
panels = (Mygroup,) # Add your panels here.
default_panel = 'mypanel' # Specify the slug of the default panel.
horizon.register(Mydashboard)
Tables, Tabs, and Views
We'll start with the table, combine that with the tabs, and then build our view from the pieces.
Defining a table
Horizon provides a ~horizon.forms.SelfHandlingForm
~horizon.tables.DataTable
class which simplifies the vast majority of displaying data to an
end-user. We're just going to skim the surface here, but it has a
tremendous number of capabilities. Create a tables.py
file
under the mypanel
directory and add the following code:
from django.utils.translation import gettext_lazy as _
from horizon import tables
class InstancesTable(tables.DataTable):
name = tables.Column("name", verbose_name=_("Name"))
status = tables.Column("status", verbose_name=_("Status"))
zone = tables.Column('availability_zone',
verbose_name=_("Availability Zone"))
image_name = tables.Column('image_name', verbose_name=_("Image Name"))
class Meta(object):
name = "instances"
verbose_name = _("Instances")
There are several things going on here... we created a table
subclass, and defined four columns that we want to retrieve data and
display. Each of those columns defines what attribute it accesses on the
instance object as the first argument, and since we like to make
everything translatable, we give each column a verbose_name
that's marked for translation.
Lastly, we added a Meta
class which indicates the meta
object that describes the instances
table.
Note
This is a slight simplification from the reality of how the instance object is actually structured. In reality, accessing other attributes requires an additional step.
Adding actions to a table
Horizon provides three types of basic action classes which can be taken on a table's data:
~horizon.tables.Action
~horizon.tables.LinkAction
~horizon.tables.FilterAction
There are also additional actions which are extensions of the basic Action classes:
~horizon.tables.BatchAction
~horizon.tables.DeleteAction
~horizon.tables.FixedFilterAction
Now let's create and add a filter action to the table. To do so, we
will need to edit the tables.py
file used above. To add a
filter action which will only show rows which contain the string entered
in the filter field, we must first define the action:
class MyFilterAction(tables.FilterAction):
name = "myfilter"
Note
The action specified above will default the filter_type
to be "query"
. This means that the filter will use the
client side table sorter.
Then, we add that action to the table actions for our table.:
class InstancesTable:
class Meta(object):
table_actions = (MyFilterAction,)
The completed tables.py
file should look like the
following:
from django.utils.translation import gettext_lazy as _
from horizon import tables
class MyFilterAction(tables.FilterAction):
name = "myfilter"
class InstancesTable(tables.DataTable):
name = tables.Column('name', \
verbose_name=_("Name"))
status = tables.Column('status', \
verbose_name=_("Status"))
zone = tables.Column('availability_zone', \
verbose_name=_("Availability Zone"))
image_name = tables.Column('image_name', \
verbose_name=_("Image Name"))
class Meta(object):
name = "instances"
verbose_name = _("Instances")
table_actions = (MyFilterAction,)
Defining tabs
So we have a table, ready to receive our data. We could go straight
to a view from here, but in this case we're also going to use horizon's
~horizon.tabs.TabGroup
class.
Create a tabs.py
file under the mypanel
directory. Let's make a tab group which has one tab. The completed code
should look like the following:
from django.utils.translation import gettext_lazy as _
from horizon import exceptions
from horizon import tabs
from openstack_dashboard import api
from openstack_dashboard.dashboards.mydashboard.mypanel import tables
class InstanceTab(tabs.TableTab):
name = _("Instances Tab")
slug = "instances_tab"
table_classes = (tables.InstancesTable,)
template_name = ("horizon/common/_detail_table.html")
preload = False
def has_more_data(self, table):
return self._has_more
def get_instances_data(self):
try:
marker = self.request.GET.get(
tables.InstancesTable._meta.pagination_param, None)
instances, self._has_more = api.nova.server_list(
self.request,
search_opts={'marker': marker, 'paginate': True})
return instances
except Exception:
self._has_more = False
error_message = _('Unable to get instances')
exceptions.handle(self.request, error_message)
return []
class MypanelTabs(tabs.TabGroup):
slug = "mypanel_tabs"
tabs = (InstanceTab,)
sticky = True
This tab gets a little more complicated. The tab handles data tables
(and all their associated features), and it also uses the
preload
attribute to specify that this tab shouldn't be
loaded by default. It will instead be loaded via AJAX when someone
clicks on it, saving us on API calls in the vast majority of cases.
Additionally, the displaying of the table is handled by a reusable
template, horizon/common/_detail_table.html
. Some simple
pagination code was added to handle large instance lists.
Lastly, this code introduces the concept of error handling in
horizon. The horizon.exceptions.handle
function is a centralized
error handling mechanism that takes all the guess-work and inconsistency
out of dealing with exceptions from the API. Use it everywhere.
Tying it together in a view
There are lots of pre-built class-based views in horizon. We try to provide the starting points for all the common combinations of components.
Open the views.py
file, the auto-generated code is like
the following:
from horizon import views
class IndexView(views.APIView):
# A very simple class-based view...
template_name = 'mydashboard/mypanel/index.html'
def get_data(self, request, context, *args, **kwargs):
# Add data to the context here...
return context
In this case we want a starting view type that works with both tabs
and tables... that'd be the ~horizon.tabs.TabbedTableView
class. It takes the
best of the dynamic delayed-loading capabilities tab groups provide and
mixes in the actions and AJAX-updating that tables are capable of with
almost no work on the user's end. Change views.APIView
to
be tabs.TabbedTableView
and add MypanelTabs
as
the tab group class in the IndexView
class:
class IndexView(tabs.TabbedTableView):
tab_group_class = mydashboard_tabs.MypanelTabs
After importing the proper package, the completed
views.py
file now looks like the following:
from horizon import tabs
from openstack_dashboard.dashboards.mydashboard.mypanel \
import tabs as mydashboard_tabs
class IndexView(tabs.TabbedTableView):
tab_group_class = mydashboard_tabs.MypanelTabs
template_name = 'mydashboard/mypanel/index.html'
def get_data(self, request, context, *args, **kwargs):
# Add data to the context here...
return context
URLs
The auto-generated urls.py
file is like:
from django.urls import re_path
from openstack_dashboard.dashboards.mydashboard.mypanel import views
urlpatterns = [
re_path(r'^$', views.IndexView.as_view(), name='index'),
]
The template
Open the index.html
file in the
mydashboard/mypanel/templates/mypanel
directory, the
auto-generated code is like the following:
{% extends 'base.html' %}
{% load i18n %}
{% block title %}{% trans "Mypanel" %}{% endblock %}
{% block page_header %}
{% include "horizon/common/_page_header.html" with title=_("Mypanel") %}
{% endblock page_header %}
{% block main %}
{% endblock %}
The main
block must be modified to insert the following
code:
<div class="row">
<div class="col-sm-12">
{{ tab_group.render }}
</div>
</div>
If you want to change the title of the index.html
file
to be something else, you can change it. For example, change it to be
My Panel
in the block title
section. If you
want the title
in the block page_header
section to be something else, you can change it. For example, change it
to be My Panel
. The updated code could be like:
{% extends 'base.html' %}
{% load i18n %}
{% block title %}{% trans "My Panel" %}{% endblock %}
{% block page_header %}
{% include "horizon/common/_page_header.html" with title=_("My Panel") %}
{% endblock page_header %}
{% block main %}
<div class="row">
<div class="col-sm-12">
{{ tab_group.render }}
</div>
</div>
{% endblock %}
This gives us a custom page title, a header, and renders our tab group provided by the view.
With all our code in place, the only thing left to do is to integrate it into our OpenStack Dashboard site.
Note
For more information about Django views, URLs and templates, please refer to the Django documentation.
Enable and show the dashboard
In order to make My Dashboard
show up along with the
existing dashboards like Project
or Admin
on
horizon, you need to create a file called
_50_mydashboard.py
under
openstack_dashboard/enabled
and add the following:
# The name of the dashboard to be added to HORIZON['dashboards']. Required.
DASHBOARD = 'mydashboard'
# If set to True, this dashboard will not be added to the settings.
DISABLED = False
# A list of applications to be added to INSTALLED_APPS.
ADD_INSTALLED_APPS = [
'openstack_dashboard.dashboards.mydashboard',
]
Run and check the dashboard
Everything is in place, now run Horizon
on the different
port:
$ tox -e runserver -- 0:9000
Go to http://<your server>:9000
using a browser.
After login as an admin you should be able see My Dashboard
shows up at the left side on horizon. Click it, My Group
will expand with My Panel
. Click on My Panel
,
the right side panel will display an Instances Tab
which
has an Instances
table.
If you don't see any instance data, you haven't created any instances
yet. Go to dashboard Project
-> Images
,
select a small image, for example, cirros-0.3.1-x86_64-uec
, click Launch
and enter an Instance Name
,
click the button Launch
. It should create an instance if
the OpenStack or devstack is correctly set up. Once the creation of an
instance is successful, go to My Dashboard
again to check
the data.
Adding a complex action to a table
For a more detailed look into adding a table action, one that
requires forms for gathering data, you can walk through tutorials-table-actions
tutorial.
Conclusion
What you've learned here is the fundamentals of how to write interfaces for your own project based on the components horizon provides.
If you have feedback on how this tutorial could be improved, please
feel free to submit a bug against horizon
.