Skip to content

Commit

Permalink
Merge pull request #3074 from digitalocean/969-custom-links
Browse files Browse the repository at this point in the history
969 custom links
  • Loading branch information
jeremystretch authored Apr 16, 2019
2 parents f342a37 + dd58e78 commit 2b2de8f
Show file tree
Hide file tree
Showing 25 changed files with 301 additions and 3 deletions.
26 changes: 25 additions & 1 deletion netbox/extras/admin.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@

from netbox.admin import admin_site
from utilities.forms import LaxURLField
from .models import CustomField, CustomFieldChoice, Graph, ExportTemplate, TopologyMap, Webhook
from .models import CustomField, CustomFieldChoice, CustomLink, Graph, ExportTemplate, TopologyMap, Webhook


def order_content_types(field):
Expand Down Expand Up @@ -77,6 +77,30 @@ def models(self, obj):
return ', '.join([ct.name for ct in obj.obj_type.all()])


#
# Custom links
#

class CustomLinkForm(forms.ModelForm):

class Meta:
model = CustomLink
exclude = []

def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)

# Format ContentType choices
order_content_types(self.fields['content_type'])
self.fields['content_type'].choices.insert(0, ('', '---------'))


@admin.register(CustomLink, site=admin_site)
class CustomLinkAdmin(admin.ModelAdmin):
list_display = ['name', 'content_type', 'group_name', 'weight']
form = CustomLinkForm


#
# Graphs
#
Expand Down
40 changes: 40 additions & 0 deletions netbox/extras/constants.py
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,46 @@
(CF_FILTER_EXACT, 'Exact'),
)

# Custom links
CUSTOM_LINK_MODELS = [
'circuits.circuit',
'circuits.provider',
'dcim.cable',
'dcim.device',
'dcim.devicetype',
'dcim.powerpanel',
'dcim.powerfeed',
'dcim.rack',
'dcim.site',
'ipam.aggregate',
'ipam.ipaddress',
'ipam.prefix',
'ipam.service',
'ipam.vlan',
'ipam.vrf',
'secrets.secret',
'tenancy.tenant',
'virtualization.cluster',
'virtualization.virtualmachine',
]

BUTTON_CLASS_DEFAULT = 'default'
BUTTON_CLASS_PRIMARY = 'primary'
BUTTON_CLASS_SUCCESS = 'success'
BUTTON_CLASS_INFO = 'info'
BUTTON_CLASS_WARNING = 'warning'
BUTTON_CLASS_DANGER = 'danger'
BUTTON_CLASS_LINK = 'link'
BUTTON_CLASS_CHOICES = (
(BUTTON_CLASS_DEFAULT, 'Default'),
(BUTTON_CLASS_PRIMARY, 'Primary (blue)'),
(BUTTON_CLASS_SUCCESS, 'Success (green)'),
(BUTTON_CLASS_INFO, 'Info (aqua)'),
(BUTTON_CLASS_WARNING, 'Warning (orange)'),
(BUTTON_CLASS_DANGER, 'Danger (red)'),
(BUTTON_CLASS_LINK, 'None (link)'),
)

# Graph types
GRAPH_TYPE_INTERFACE = 100
GRAPH_TYPE_PROVIDER = 200
Expand Down
32 changes: 32 additions & 0 deletions netbox/extras/migrations/0022_custom_links.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
# Generated by Django 2.2 on 2019-04-15 19:28

from django.db import migrations, models
import django.db.models.deletion


class Migration(migrations.Migration):

dependencies = [
('contenttypes', '0002_remove_content_type_name'),
('extras', '0021_add_color_comments_changelog_to_tag'),
]

operations = [
migrations.CreateModel(
name='CustomLink',
fields=[
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False)),
('name', models.CharField(max_length=100, unique=True)),
('text', models.CharField(max_length=200)),
('url', models.CharField(max_length=200)),
('weight', models.PositiveSmallIntegerField(default=100)),
('group_name', models.CharField(blank=True, max_length=50)),
('button_class', models.CharField(default='default', max_length=30)),
('new_window', models.BooleanField()),
('content_type', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='contenttypes.ContentType')),
],
options={
'ordering': ['group_name', 'weight', 'name'],
},
),
]
59 changes: 59 additions & 0 deletions netbox/extras/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -300,6 +300,65 @@ def delete(self, using=None, keep_parents=False):
CustomFieldValue.objects.filter(field__type=CF_TYPE_SELECT, serialized_value=str(pk)).delete()


#
# Custom links
#

def get_custom_link_models():
# TODO: This should match on the app_label as well as the model name to avoid potential duplicate names
return {
'model__in': [model.split('.')[1] for model in CUSTOM_LINK_MODELS],
}


class CustomLink(models.Model):
"""
A custom link to an external representation of a NetBox object. The link text and URL fields accept Jinja2 template
code to be rendered with an object as context.
"""
content_type = models.ForeignKey(
to=ContentType,
on_delete=models.CASCADE,
limit_choices_to=get_custom_link_models
)
name = models.CharField(
max_length=100,
unique=True
)
text = models.CharField(
max_length=200,
help_text="Jinja2 template code for link text"
)
url = models.CharField(
max_length=200,
verbose_name='URL',
help_text="Jinja2 template code for link URL"
)
weight = models.PositiveSmallIntegerField(
default=100
)
group_name = models.CharField(
max_length=50,
blank=True,
help_text="Links with the same group will appear as a dropdown menu"
)
button_class = models.CharField(
max_length=30,
choices=BUTTON_CLASS_CHOICES,
default=BUTTON_CLASS_DEFAULT,
help_text="The class of the first link in a group will be used for the dropdown button"
)
new_window = models.BooleanField(
help_text="Force link to open in a new window"
)

class Meta:
ordering = ['group_name', 'weight', 'name']

def __str__(self):
return self.name


#
# Graphs
#
Expand Down
Empty file.
68 changes: 68 additions & 0 deletions netbox/extras/templatetags/custom_links.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
from collections import OrderedDict

from django import template
from django.contrib.contenttypes.models import ContentType
from django.utils.safestring import mark_safe
from jinja2 import Environment

from extras.models import CustomLink


register = template.Library()

LINK_BUTTON = '<a href="{}"{} class="btn btn-sm btn-{}">{}</a>\n'
GROUP_BUTTON = '<div class="btn-group">\n' \
'<button type="button" class="btn btn-sm btn-{} dropdown-toggle" data-toggle="dropdown">\n' \
'{} <span class="caret"></span>\n' \
'</button>\n' \
'<ul class="dropdown-menu pull-right">\n'
GROUP_LINK = '<li><a href="{}"{}>{}</a></li>\n'


@register.simple_tag()
def custom_links(obj):
"""
Render all applicable links for the given object.
"""
content_type = ContentType.objects.get_for_model(obj)
custom_links = CustomLink.objects.filter(content_type=content_type)
if not custom_links:
return ''

context = {
'obj': obj,
}
template_code = ''
group_names = OrderedDict()

# Organize custom links by group
for cl in custom_links:
if cl.group_name and cl.group_name in group_names:
group_names[cl.group_name].append(cl)
elif cl.group_name:
group_names[cl.group_name] = [cl]

# Add non-grouped links
for cl in custom_links:
if not cl.group_name:
link_target = ' target="_blank"' if cl.new_window else ''
template_code += LINK_BUTTON.format(
cl.url, link_target, cl.button_class, cl.text
)

# Add grouped links to template
for group, links in group_names.items():
template_code += GROUP_BUTTON.format(
links[0].button_class, group
)
for cl in links:
link_target = ' target="_blank"' if cl.new_window else ''
template_code += GROUP_LINK.format(
cl.url, link_target, cl.text
)
template_code += '</ul>\n</div>\n'

# Render template
rendered = Environment().from_string(source=template_code).render(**context)

return mark_safe(rendered)
4 changes: 4 additions & 0 deletions netbox/templates/circuits/circuit.html
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
{% extends '_base.html' %}
{% load custom_links %}
{% load helpers %}

{% block title %}{{ circuit }}{% endblock %}
Expand Down Expand Up @@ -41,6 +42,9 @@
</div>
<h1>{{ circuit }}</h1>
{% include 'inc/created_updated.html' with obj=circuit %}
<div class="pull-right noprint">
{% custom_links circuit %}
</div>
<ul class="nav nav-tabs">
<li role="presentation"{% if not active_tab %} class="active"{% endif %}>
<a href="{{ circuit.get_absolute_url }}">Circuit</a>
Expand Down
4 changes: 4 additions & 0 deletions netbox/templates/circuits/provider.html
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
{% extends '_base.html' %}
{% load static %}
{% load custom_links %}
{% load helpers %}

{% block title %}{{ provider }}{% endblock %}
Expand Down Expand Up @@ -47,6 +48,9 @@
</div>
<h1>{{ provider }}</h1>
{% include 'inc/created_updated.html' with obj=provider %}
<div class="pull-right noprint">
{% custom_links provider %}
</div>
<ul class="nav nav-tabs">
<li role="presentation"{% if not active_tab %} class="active"{% endif %}>
<a href="{{ provider.get_absolute_url }}">Provider</a>
Expand Down
5 changes: 5 additions & 0 deletions netbox/templates/dcim/cable.html
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
{% extends '_base.html' %}
{% load custom_links %}
{% load helpers %}

{% block header %}
Expand All @@ -23,6 +24,10 @@
{% endif %}
</div>
<h1>{% block title %}Cable {{ cable }}{% endblock %}</h1>
{% include 'inc/created_updated.html' with obj=cable %}
<div class="pull-right noprint">
{% custom_links cable %}
</div>
<ul class="nav nav-tabs">
<li role="presentation"{% if not active_tab %} class="active"{% endif %}>
<a href="{{ cable.get_absolute_url }}">Cable</a>
Expand Down
4 changes: 4 additions & 0 deletions netbox/templates/dcim/device.html
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
{% extends '_base.html' %}
{% load static %}
{% load helpers %}
{% load custom_links %}

{% block title %}{{ device }}{% endblock %}

Expand Down Expand Up @@ -64,6 +65,9 @@
</div>
<h1>{{ device }}</h1>
{% include 'inc/created_updated.html' with obj=device %}
<div class="pull-right noprint">
{% custom_links device %}
</div>
<ul class="nav nav-tabs">
<li role="presentation"{% if not active_tab %} class="active"{% endif %}>
<a href="{% url 'dcim:device' pk=device.pk %}">Device</a>
Expand Down
4 changes: 4 additions & 0 deletions netbox/templates/dcim/devicetype.html
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
{% extends '_base.html' %}
{% load custom_links %}
{% load helpers %}

{% block title %}{{ devicetype.manufacturer }} {{ devicetype.model }}{% endblock %}
Expand Down Expand Up @@ -46,6 +47,9 @@
{% endif %}
<h1>{{ devicetype.manufacturer }} {{ devicetype.model }}</h1>
{% include 'inc/created_updated.html' with obj=devicetype %}
<div class="pull-right noprint">
{% custom_links devicetype %}
</div>
<ul class="nav nav-tabs">
<li role="presentation"{% if not active_tab %} class="active"{% endif %}>
<a href="{{ devicetype.get_absolute_url }}">Device Type</a>
Expand Down
5 changes: 4 additions & 1 deletion netbox/templates/dcim/powerfeed.html
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{% extends '_base.html' %}
{% load static %}
{% load tz %}
{% load custom_links %}
{% load helpers %}

{% block header %}
Expand Down Expand Up @@ -45,6 +45,9 @@
</div>
<h1>{% block title %}{{ powerfeed }}{% endblock %}</h1>
{% include 'inc/created_updated.html' with obj=powerfeed %}
<div class="pull-right noprint">
{% custom_links powerfeed %}
</div>
<ul class="nav nav-tabs">
<li role="presentation"{% if not active_tab %} class="active"{% endif %}>
<a href="{{ powerfeed.get_absolute_url }}">Cable</a>
Expand Down
5 changes: 4 additions & 1 deletion netbox/templates/dcim/powerpanel.html
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{% extends '_base.html' %}
{% load static %}
{% load tz %}
{% load custom_links %}
{% load helpers %}

{% block header %}
Expand Down Expand Up @@ -44,6 +44,9 @@
</div>
<h1>{% block title %}{{ powerpanel }}{% endblock %}</h1>
{% include 'inc/created_updated.html' with obj=powerpanel %}
<div class="pull-right noprint">
{% custom_links powerpanel %}
</div>
<ul class="nav nav-tabs">
<li role="presentation"{% if not active_tab %} class="active"{% endif %}>
<a href="{{ powerpanel.get_absolute_url }}">Cable</a>
Expand Down
4 changes: 4 additions & 0 deletions netbox/templates/dcim/rack.html
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
{% extends '_base.html' %}
{% load custom_links %}
{% load helpers %}

{% block header %}
Expand Down Expand Up @@ -43,6 +44,9 @@
</div>
<h1>{% block title %}Rack {{ rack }}{% endblock %}</h1>
{% include 'inc/created_updated.html' with obj=rack %}
<div class="pull-right noprint">
{% custom_links rack %}
</div>
<ul class="nav nav-tabs">
<li role="presentation"{% if not active_tab %} class="active"{% endif %}>
<a href="{{ rack.get_absolute_url }}">Rack</a>
Expand Down
Loading

0 comments on commit 2b2de8f

Please sign in to comment.