Skip to main content

Customisable, modular dashboard application framework for Django.

Project description

django-dash (later on named Dash) is a customisable, modular dashboard application framework for Django.

Dash allows users to create their own custom dashboards. Supports theming (in Dash themes are called layouts) and multiple workspaces. Dash comes with extensive pythonic API which allows developers to create new Dash plugins, as well as to modify bundled ones.

To make a clearer association, think of Android for tablets (shortcuts, widgets and apps) or Windows 8 for tablets or desktops.

Dash inherits all those concepts and makes it possible to implement a dashboard system for Django applications with minimal efforts.

Prerequisites

  • Django 1.5.+

  • Python 2.6.8+, 2.7.+, 3.3.+

Key concepts

  • Each layout (theme) consist of placeholders. Each plugin widget has its’ own specific HTML/JavaScript/CSS.

  • There might be multiple themes implemented and installed, but only one can be active for a certain user. Default layout is chosen system wide, but each user (if has an appropriate permission) can choose his preferred layout.

  • Placeholder is a space, in which the plugin widgets are placed.

  • Placeholders are rectangles consisting of cells. Each placeholder has its’ own custom number of rows and columns.

  • Workspace is just another named dashboard. Users switch between workspaces in navigation. Amount of workspaces is unlimited.

  • Plugin is a (Django) micro app. Most heavy work should happen in plugin. Plugin may have its’ own views, urls, etc. Rendering happens with use of plugin widgets.

  • Plugin widgets are mainly responsible for rendering of the plugin data. Each plugin widget has its’ own specific HTML/JavaScript/CSS. A single plugin widget is registered for a triple (layout, placeholder, plugin).

  • Public dashboard (implemented as a contrib app, which makes it optional) allows users to make their workspaces public. If user chooses to make his dashboard public, default workspace becomes public. As for non-default workspaces, user can still make each of them private or public.

Main features

  • Customisable layouts (aka theeming).

  • Multiple workspaces.

  • Tunable access permissions to plugins.

  • Public dashboards (as a contrib app).

Installation

  1. Install latest stable version from PyPI:

    $ pip install django-dash

    Or latest stable version from GitHub:

    $ pip install -e git+https://github.com/barseghyanartur/django-dash@stable#egg=django-dash

    Or latest stable version from BitBucket:

    $ pip install -e hg+https://bitbucket.org/barseghyanartur/django-dash@stable#egg=django-dash

  2. Add dash to INSTALLED_APPS of the your projects’ Django settings. Furthermore, all layouts and plugins to be used, shall be added to the INSTALLED_APPS as well.

>>> INSTALLED_APPS = (
>>>     # ...
>>>     'dash',
>>>     'dash.contrib.layouts.android',
>>>     'dash.contrib.layouts.windows8',
>>>     'dash.contrib.plugins.dummy',
>>>     'dash.contrib.plugins.image',
>>>     'dash.contrib.plugins.memo',
>>>     'dash.contrib.plugins.news',
>>>     'dash.contrib.plugins.rss_feed',
>>>     'dash.contrib.plugins.url',
>>>     'dash.contrib.plugins.video',
>>>     'dash.contrib.plugins.weather',
>>>     # ...
>>> )
  1. Make sure that django.core.context_processors.request is in TEMPLATE_CONTEXT_PROCESSORS.

Demo

In order to be able to quickly evaluate the django-dash, a demo app (with a quick installer) has been created (works on Ubuntu/Debian, may work on other Linux systems as well, although not guaranteed). Follow the instructions below for having the demo running within a minute.

Grab the latest django_dash_example_app_installer.sh:

$ wget https://raw.github.com/barseghyanartur/django-dash/stable/example/django_dash_example_app_installer.sh

Create a new- or switch to existing- virtual environement:

$ virtualenv dash

$ source dash/bin/activate

Assign execute rights to the installer and run the django_dash_example_app_installer.sh:

$ chmod +x django_dash_example_app_installer.sh

$ ./django_dash_example_app_installer.sh

Open your browser and test the app.

Dashboard:

Django admin interface:

Take a look at the templates in “example/example/templates” directory for getting a better idea of how to transform your own- or thirdy-part- templates into Dash templates.

See the documentation for some screen shots http://pythonhosted.org/django-dash/#screenshots

Creating a new layout

Dash comes with several bundled layouts. Do check their source code as example.

Let’s say, our imaginary layout has two placeholders. One large placeholder for all kinds of widgets (called main) and a tiny one for shortcuts (called shortcuts).

Placeholder main:

  • Single cell size : 150 x 110 pixels

  • Dimensions : 6 cols, 5 rows

Placeholder shortcuts:

  • Single cell size : 60 x 55 pixels

  • Dimensions : 1 cols, 10 rows

See the figure below to get an idea of what placeholders are:

  • Placeholder main consts of cells from 11 to 56.

  • Placeholder shortcuts consists of cells from 1 to 10.

A single plugin widget may occupy one or more cells. Plugin widgets are rectangles.

To make it clear, see following cases:

  • Plugin widget has 2 cols and 1 row. Then, for example, it may occupy cells (11 and 12).

  • Plugin widget has 2 cols and 2 rows. Then, for example, it may occupy cells (11, 12, 21 and 22).

  • Plugin widget has 1 col and 3 rows. Then, for example, it may occupy cells (11, 21 and 31).

  • Plugin widget has 4 cols and 3 rows. Then, for example, it may occupy cells (22, 23, 24, 25, 32, 33, 34, 35, 42, 43, 44 and 45).

>>>                                  `main`                                `shortcuts`
>>>┌───────────┬───────────┬───────────┬───────────┬───────────┬───────────┐ ┌─────┐
>>>│           │           │           │           │           │           │ │  1  │
>>>│           │           │           │           │           │           │ │     │
>>>│    11     │    12     │    13     │    14     │    15     │    16     │ ├─────┤
>>>│           │           │           │           │           │           │ │  2  │
>>>│           │           │           │           │           │           │ │     │
>>>├───────────┼───────────┼───────────┼───────────┼───────────┼───────────┤ ├─────┤
>>>│           │           │           │           │           │           │ │     │
>>>│           │           │           │           │           │           │ │  3  │
>>>│    21     │    22     │    23     │    24     │    25     │    26     │ ├─────┤
>>>│           │           │           │           │           │           │ │  4  │
>>>│           │           │           │           │           │           │ │     │
>>>├───────────┼───────────┼───────────┼───────────┼───────────┼───────────┤ ├─────┤
>>>│           │           │           │           │           │           │ │     │
>>>│           │           │           │           │           │           │ │  5  │
>>>│    31     │    32     │    33     │    34     │    35     │    36     │ ├─────┤
>>>│           │           │           │           │           │           │ │  6  │
>>>│           │           │           │           │           │           │ │     │
>>>├───────────┼───────────┼───────────┼───────────┼───────────┼───────────┤ ├─────┤
>>>│           │           │           │           │           │           │ │     │
>>>│           │           │           │           │           │           │ │  7  │
>>>│    41     │    42     │    43     │    44     │    45     │    46     │ ├─────┤
>>>│           │           │           │           │           │           │ │  8  │
>>>│           │           │           │           │           │           │ │     │
>>>├───────────┼───────────┼───────────┼───────────┼───────────┼───────────┤ ├─────┤
>>>│           │           │           │           │           │           │ │     │
>>>│           │           │           │           │           │           │ │  9  │
>>>│    51     │    52     │    53     │    54     │    55     │    56     │ ├─────┤
>>>│           │           │           │           │           │           │ │ 10  │
>>>│           │           │           │           │           │           │ │     │
>>>└───────────┴───────────┴───────────┴───────────┴───────────┴───────────┘ └─────┘

There are some rules/guideles you should follow.

Let’s assume that layout is named example. The layout directory should then have the following structure.

>>> path/to/layout/example/
>>> ├── static
>>> │   ├── css
>>> │   │   └── dash_layout_example.css # Contains layout-specific CSS
>>> │   ├── images
>>> │   └── js
>>> │       └── dash_layout_example.js # Contains layout specific JavaScripts
>>> ├── templates
>>> │   └── example
>>> │       ├── edit_layout.html # Master edit layout
>>> │       └── view_layout.html # Master view layout
>>> ├── __init__.py
>>> └── dash_layouts.py # Where Layouts and Placeholders are defined and registered

Layout and placeholder classes should be placed in the dash_layouts.py file.

Each layout should be put into the INSTALLED_APPS of your Django projects’ settings module.

>>> INSTALLED_APPS = (
>>>     # ...
>>>     'path.to.layout.example',
>>>     # ...
>>> )

path/to/layout/example/dash_layouts.py

Step by step review of a how to create and register a layout and placeholders. Note, that Dash autodiscovers your layouts by name of the file dash_layouts.py. The module, in which the layouts are defined, has to be named dash_layouts.py.

Required imports.

>>> from dash.base import BaseDashboardLayout, BaseDashboardPlaceholder
>>> from dash.base import layout_registry

Defining the Main placeholder.

>>> class ExampleMainPlaceholder(BaseDashboardPlaceholder):
>>>    uid = 'main' # Unique ID of the placeholder.
>>>    cols = 6 # Number of columns in the placeholder.
>>>    rows = 5 # Number of rows in the placeholder.
>>>    cell_width = 150 # Width of a single cell in the placeholder.
>>>    cell_height = 110 # Height of a single cell in the placeholder.

Defining the Shortcuts placeholder.

>>> class ExampleShortcutsPlaceholder(BaseDashboardPlaceholder):
>>>     uid = 'shortcuts' # UID of the placeholder.
>>>     cols = 1 # Number of columns in the placeholder.
>>>     rows = 10 # Number of rows in the placeholder.
>>>     cell_width = 60 # Width of a single cell in the placeholder.
>>>     cell_height = 55 # Height of a single cell in the placeholder.

Defining and registering the Layout.

>>> class ExampleLayout(BaseDashboardLayout):
>>>     uid = 'example' # Layout UID.
>>>     name = 'Example' # Layout name.
>>>
>>>     # View template. Master template used in view mode.
>>>     view_template_name = 'example/view_layout.html'
>>>
>>>     # Edit template. Master template used in edit mode.
>>>     edit_template_name = 'example/edit_layout.html'
>>>
>>>     # All placeholders listed. Note, that placeholders are rendered in the
>>>     # order specified here.
>>>     placeholders = [ExampleMainPlaceholder, ExampleShortcutsPlaceholder]
>>>
>>>     # Cell units used in the entire layout. Allowed values are: 'px', 'pt',
>>>     # 'em' or '%'. In the ``ExampleMainPlaceholder`` cell_width is set to 150.
>>>     #  It means that in this particular case its' actual width would be `150px`.
>>>     cell_units = 'px'
>>>
>>>     # Layout specific CSS.
>>>     media_css = ('css/dash_layout_example.css',)
>>>
>>>     # Layout specific JS.
>>>     media_js = ('js/dash_layout_example.js',)
>>>
>>> # Registering the layout.
>>> layout_registry.register(ExampleLayout)

HTML templates

You custom layout should be interited from base layout templates (view or edit). Both view and edit layouts share a lot of things, still edit layout is a bit more “heavy”.

  • view_layout.html should inherit from “dash/layouts/base_view_layout.html”.

  • edit_layout.html should inherit from “dash/layouts/base_edit_layout.html”.

Both “dash/layouts/base_view_layout.html” and “dash/layouts/base_edit_layout.html” inherit from “dash/layouts/base_layout.html”, which in its’ turn inherits from “dash/base.html”.

Note, that when rendered to HTML, each Dash template, gets a body class “layout” + layouts’ unique identifier (UID). So, the ExampleLayout layout would automatically get the class “layout-example”.

<body class=”layout-example”>

In case of Android layout (UID “android”) it would be as follows.

<body class=”layout-android”>

Base your layout specific custom CSS on presence of those classes.

Same goes for Placeholders. Each placeholder gets id_ + placeholders’ UID and the classes “placeholder” and “placeholder-” + placeholders’ UID. So, the ExampleMainPlaceholder would look as follows.

<div id=”id_main” class=”placeholder placeholder-main”>

And the ExampleShortcutsPlaceholder placeholder would look as follows.

<div id=”id_shortcuts” class=”placeholder placeholder-shortcuts”>

Same goes for plugin widgets. Apart from some other classes that each plugin widget would get for positioning, it gets the “plugin” and “plugin-” + plugin UID. See the following example (for the plugin Dummy with UID “dummy”). Each plugin also gets an automatic UID on the moment when rendered. In the example below it’s the “p6d06f17d-e142-4f45-b9c1-893c38fc2b01”.

<div id=”p6d06f17d-e142-4f45-b9c1-893c38fc2b01” class=”plugin plugin-dummy”>

Layout, placeholder, plugin and plugin widget have properties for getting their HTML specific classes and IDs.

Layout (instance)

>>> layout.html_class

Placeholder (instance)

>>> placeholder.html_id
>>> placeholder.html_class

Plugin (instance)

>>> plugin.html_id
>>> plugin.html_class

Plugin widget (static call)

>>> plugin_widget.html_class # Static one

Creating a new plugin

Dash comes with several bundled plugins. Do check their source code as example.

Making of a plugin or a plugin widget is quite simple, although there are some rules/guideles you should follow.

Let’s assume that plugin is named sample_memo. The plugin directory should then have the following structure.

Note, that you are advised to prefix all your plugin specific media files with dash_plugin_ for the sake of common sense.

>>> path/to/plugin/sample_memo/
>>> ├── static
>>> │   ├── css
>>> │   │   └── dash_plugin_sample_memo.css # Plugin specific CSS
>>> │   ├── images
>>> │   └── js
>>> │       └── dash_plugin_sample_memo.js # Plugin specific JavaScripts
>>> ├── templates
>>> │   └── sample_memo
>>> │       ├── render_main.html # Plugin widget templ. for `main` Placeholder
>>> │       └── render_short.html # Plugin widget templ. for `shortcuts` Placeholder
>>> ├── __init__.py
>>> ├── dash_plugins.py # Where plugins and widgets are defined and registered
>>> ├── dash_widgets.py # Where the plugin widgets are defined
>>> └── forms.py # Plugin configuration form

In some cases, you would need plugin specific overridable settings (see dash.contrib.plugins.weather plugin as an example. You are advised to write your settings in such a way, that variables of your Django project settings module would have DASH_PLUGIN_ prefix.

path/to/plugin/sample_memo/dash_plugins.py

Step by step review of a how to create and register a plugin and plugin widgets. Note, that Dash autodiscovers your plugins if you place them into a file named dash_plugins.py of any Django app listed in INSTALLED_APPS of your Django projects’ settings module.

Define and register the plugin

Required imports.

>>> from dash.base import BaseDashboardPlugin, plugin_registry
>>> from path.to.plugin.sample_memo.forms import SampleMemoForm

Defining the Sample Memo plugin.

>>> class SampleMemoPlugin(BaseDashboardPlugin):
>>>     uid = 'sample_memo' # Plugin UID
>>>     name = _("Memo") # Plugin name
>>>     group = _("Memo") # Group to which the plugin belongs to
>>>     form = SampleMemoForm # Plugin forms are explained later

Registering the Sample Memo plugin.

>>> plugin_registry.register(SampleMemoPlugin)

Register plugin widgets

Plugin widgets are defined in dash_widgets.py module (described later), but registered in the dash_plugins.py, which is autodiscovered by Dash.

Required imports.

>>> from dash.base import plugin_widget_registry
>>> from path.to.plugin.sample_memo.dash_widgets import (
>>>     SampleMemoExampleMainWidget, SampleMemoExampleShortcutWidget
>>>     )

Registering the Sample Memo plugin widgets for Layout example.

>>> plugin_widget_registry.register(SampleMemoExampleMainWidget)
>>> plugin_widget_registry.register(SampleMemoExampleShortcutWidget)

path/to/plugin/sample_memo/dash_widgets.py

Why to have another file for defining widgets? Just to keep the code clean and less messy, although you could perfectly define all your plugin widgets in the module dash_plugins.py, it’s recommended to keep it separate.

Take into consideration, that dash_widgets.py is not an autodiscovered file pattern. All your plugin widgets should be registered in modules named dash_plugins.py.

Required imports.

>>> from django.template.loader import render_to_string
>>> from dash.base import BaseDashboardPluginWidget

Memo plugin widget for Example layout (Placeholder main).

>>> class SampleExampleMemoExampleMainWidget(BaseDashboardPluginWidget):
>>>     layout_uid = 'example' # Layout for which the widget is written
>>>     placeholder_uid = 'main' # Placeholder within the layout for which
>>>                              # the widget is written
>>>     plugin_uid = 'sample_memo' # Plugin for which the widget is written
>>>     cols = 2 # Number of widget columns
>>>     rows = 2 # Number of widget rows
>>>
>>>     def render(self, request=None):
>>>         context = {'plugin': self.plugin}
>>>         return render_to_string('sample_memo/render_main.html', context)

Memo plugin widget for Example layout (Placeholder shortcuts).

>>> class SampleMemoExampleShortcutWidget(SampleMemoExampleMainWidget):
>>>     placeholder_uid = 'shortcuts'
>>>     cols = 1
>>>     rows = 1
>>>
>>>     def render(self, request=None):
>>>         context = {'plugin': self.plugin}
>>>         return render_to_string('sample_memo/render_shortcuts.html', context)

path/to/plugin/sample_memo/forms.py

What are the plugin forms? Very simple - if plugin is configurable, it has a form. If you need to have a custom CSS or a JavaScript included when rendering a speicifc form, use Django’s class Media directive in the form.

Required imports.

>>> from django import forms
>>> from dash.base import DashboardPluginFormBase

Memo form (for Sample Memo plugin).

>>> class SampleMemoForm(forms.Form, DashboardPluginFormBase):
>>>     plugin_data_fields = [
>>>         ("title", ""),
>>>         ("text", "")
>>>     ]
>>>
>>>     title = forms.CharField(label=_("Title"), required=False)
>>>     text = forms.CharField(label=_("Text"), required=True, \
>>>                            widget=forms.widgets.Textarea)
>>>
>>>     def __init__(self, *args, **kwargs):
>>>         super(MemoForm, self).__init__(*args, **kwargs)

Now, that everything is ready, make sure your that both layout and the plugin modules are added to INSTALLED_APPS for your projects’ Django settings.

>>> INSTALLED_APPS = (
>>>     # ...
>>>     'path.to.layout.example',
>>>     'path.to.plugin.sample_memo',
>>>     # ...
>>> )

After it’s done, go to terminal and type the following command.

$ ./manage.py dash_sync_plugins

If your HTTP server is running, you would then be able to access your dashboard.

Note, that you have to be logged in, in order to use the dashboard. If your new plugin doesn’t appear, set the DASH_DEBUG to True in your Django’s local settings module, re-run your code and check console for error notifications.

Permissions

Plugin system allows administrators to specify the access rights to every plugin. Dash permissions are based on Django Users and User Groups. Access rights are managable via Django admin (/administration/dash/dashboardplugin/). Note, that your admin URL prefix may vary from the one given in example (it’s usually “/admin/”, while in example it’s “/administration/”). If user doesn’t have the rights to access plugin, it doesn’t appear on his dashboard even if has been added to it (imagine, you have once granted the right to use the news plugin to all users, but later on decided to limit it to Staff members group only). Note, that superusers have access to all plugins.

>>>         Plugin access rights management interface in Django admin
>>>┌──────────────────────────────┬───────────────────────┬───────────────────────┐
>>>│ `Plugin`                     │ `Users`               │ `Groups`              │
>>>├──────────────────────────────┼───────────────────────┼───────────────────────┤
>>>│ Video (big_video)            │ John Doe              │ Dashboard users       │
>>>├──────────────────────────────┼───────────────────────┼───────────────────────┤
>>>│ TinyMCE memo (tinymce_memo)  │                       │ Dashboard users       │
>>>├──────────────────────────────┼───────────────────────┼───────────────────────┤
>>>│ News (news)                  │ Oscar, John Doe       │ Staff members         │
>>>├──────────────────────────────┼───────────────────────┼───────────────────────┤
>>>│ URL (url)                    │                       │ Dashboard users       │
>>>├──────────────────────────────┼───────────────────────┼───────────────────────┤
>>>│ Video (video)                │                       │ Dashboard users       │
>>>├──────────────────────────────┼───────────────────────┼───────────────────────┤
>>>│ Dummy (dummy)                │                       │ Testers               │
>>>├──────────────────────────────┼───────────────────────┼───────────────────────┤
>>>│ Dummy (large_dummy)          │                       │ Testers               │
>>>├──────────────────────────────┼───────────────────────┼───────────────────────┤
>>>│ Memo (big_memo)              │                       │ Dashboard users       │
>>>└──────────────────────────────┴───────────────────────┴───────────────────────┘

Management commands

There are several management commands.

  • dash_find_broken_dashboard_entries. Find broken dashboard entries that occur when some plugin which did exist in the system, no longer exists.

  • dash_sync_plugins. Should be ran each time a new plugin is being added to the Dash.

  • dash_update_plugin_data. A mechanism to update existing plugin data in case if it had become invalid after a change in a plugin. In order for it to work, each plugin should implement and update method, in which the data update happens.

Tuning

There are number of Dash settings you can override in the settings module of your Django project:

  • DASH_RESTRICT_PLUGIN_ACCESS (bool): If set to True, (Django) permission system for dash plugins is enabled. Defaults to True. Setting this to False makes all plugins available for all users.

  • DASH_ACTIVE_LAYOUT (str): Active (default) layout UID. Defaults to “android”.

  • DASH_LAYOUT_CELL_UNITS (str): Allowed values for layout cell units. Defaults to (“em”, “px”, “pt”, “%”).

  • DASH_DISPLAY_AUTH_LINK (bool): If set to True, the log out link is shown in the Dash drop-down menu. Defaults to True.

For tuning of specific contrib plugin, see the docs in the plugin directory.

Styling tips

Font Awesome is used for icons. As a convension, all icons of font-awesome are placed within a span. Next to their original class, they all should be getting an extra class “iconic”. Follow that rule when making a new layout or a plugin (HTML). It allows to make the styling easy, since icon colours could be then changed within no time.

Bundled plugins and layouts

Dash ships with number of bundled (demo) plugins and layouts that are mainly made to demonstrate its’ abilities. In order to work amoung various layouts (themes), each plugin has a single widget registered for a single layout. It’s possible to unregister a bundled widget and replace it with a custom one.

Bundled plugins

Below a short overview of the plugins. See the README.rst file in directory of each plugin for details.

Bundled layouts

Below a short overview of the layouts. See the README.rst file in directory of each layout for details.

Naming conventions

Although you are free to name your plugins and widgets as you want (except that you should comply with PEP http://www.python.org/dev/peps/pep-0008/#function-names), there are some naming conventions introduced, that you are recommended to follow.

  • Example1x1Plugin: 1x1 example plugin
    • Example1x1AndroidMainWidget: 1x1 widget for 1x1 example plugin (layout Android, placeholder ‘main’)

    • Example1x1AndroidShortcutsWidget: 1x1 widget for 1x1 example plugin (layout Android, placeholder ‘shortcuts’)

    • Example1x1Windows8MainWidget: 1x1 widget for 1x1 example plugin (layout Windows 8, placeholder ‘main’)

    • Example1x1Windows8SidebarWidget: 1x1 widget for 1x1 example plugin (layout Windows 8, placeholder ‘sidebar’)

  • Example2x3Plugin: 2x3 example plugin
    • Example2x3Windows8MainWidget: 2x3 widget for 2x3 example plugin (layout Windows 8, placeholder ‘main’)

    • Example2x3Windows8SidebarWidget: 2x3 widget for 2x3 example plugin (layout Windows 8, placeholder ‘sidebar’)

  • Example6x1Plugin: 6x1 example plugin
    • Example6x1YourLayoutSidebarWidget: 6x1 widget for 6x1 example plugin (layout Your Layout, placeholder ‘main’)

License

GPL 2.0/LGPL 2.1

Support

For any issues contact me at the e-mail given in the Author section.

Author

Artur Barseghyan <artur.barseghyan@gmail.com>

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

django-dash-0.2.1.tar.gz (1.9 MB view details)

Uploaded Source

File details

Details for the file django-dash-0.2.1.tar.gz.

File metadata

File hashes

Hashes for django-dash-0.2.1.tar.gz
Algorithm Hash digest
SHA256 611bc3051b5c4f6ea40cdf533b9ee9c7a0861945d1b6088b48ef051914ba5bc5
MD5 da0928075e158cd538d4d3019d685dfc
BLAKE2b-256 4885b3da89f0d674652dac525c3dbcfb41607667fa15649f74ffcd86e758da17

See more details on using hashes here.

Provenance

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page