nautobot 2.4.17__py3-none-any.whl → 2.4.18__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of nautobot might be problematic. Click here for more details.
- nautobot/apps/views.py +2 -0
- nautobot/circuits/templates/circuits/circuittermination_retrieve.html +1 -8
- nautobot/circuits/templates/circuits/inc/circuit_termination_speed_fragment.html +9 -0
- nautobot/circuits/tests/integration/test_circuit.py +2 -2
- nautobot/circuits/views.py +32 -15
- nautobot/core/filters.py +2 -2
- nautobot/core/settings.py +1 -0
- nautobot/core/settings.yaml +9 -0
- nautobot/core/tables.py +21 -23
- nautobot/core/templates/components/breadcrumbs.html +19 -0
- nautobot/core/templates/generic/object_changelog.html +0 -2
- nautobot/core/templates/generic/object_list.html +15 -12
- nautobot/core/templates/generic/object_notes.html +0 -2
- nautobot/core/templates/generic/object_retrieve.html +16 -9
- nautobot/core/templatetags/helpers.py +24 -0
- nautobot/core/templatetags/ui_framework.py +40 -5
- nautobot/core/testing/filters.py +37 -21
- nautobot/core/testing/views.py +25 -0
- nautobot/core/tests/test_tables.py +43 -6
- nautobot/core/tests/test_templatetags_ui_framework.py +146 -0
- nautobot/core/tests/test_titles.py +2 -2
- nautobot/core/tests/test_ui.py +14 -1
- nautobot/core/tests/test_views.py +45 -0
- nautobot/core/ui/breadcrumbs.py +13 -8
- nautobot/core/ui/object_detail.py +43 -5
- nautobot/core/ui/titles.py +9 -5
- nautobot/core/views/__init__.py +24 -3
- nautobot/core/views/generic.py +42 -17
- nautobot/core/views/mixins.py +146 -12
- nautobot/core/views/utils.py +117 -0
- nautobot/dcim/models/devices.py +4 -0
- nautobot/dcim/tables/__init__.py +2 -0
- nautobot/dcim/tables/devices.py +24 -0
- nautobot/dcim/tables/power.py +2 -2
- nautobot/dcim/templates/dcim/device/base.html +1 -11
- nautobot/dcim/templates/dcim/device_component.html +0 -19
- nautobot/dcim/templates/dcim/modulebay_retrieve.html +0 -16
- nautobot/dcim/templates/dcim/virtualchassis_retrieve.html +1 -50
- nautobot/dcim/tests/test_views.py +41 -0
- nautobot/dcim/views.py +160 -39
- nautobot/extras/filters/mixins.py +1 -1
- nautobot/extras/forms/forms.py +15 -0
- nautobot/extras/models/groups.py +10 -1
- nautobot/extras/models/jobs.py +2 -2
- nautobot/extras/plugins/views.py +18 -5
- nautobot/extras/tables.py +4 -2
- nautobot/extras/templates/extras/customfield_retrieve.html +1 -128
- nautobot/extras/templates/extras/dynamicgroup.html +2 -99
- nautobot/extras/templates/extras/dynamicgroup_edit.html +2 -199
- nautobot/extras/templates/extras/dynamicgroup_retrieve.html +99 -0
- nautobot/extras/templates/extras/dynamicgroup_update.html +199 -0
- nautobot/extras/templates/extras/gitrepository.html +2 -82
- nautobot/extras/templates/extras/gitrepository_object_edit.html +2 -13
- nautobot/extras/templates/extras/gitrepository_retrieve.html +82 -0
- nautobot/extras/templates/extras/gitrepository_update.html +13 -0
- nautobot/extras/templates/extras/note_retrieve.html +0 -52
- nautobot/extras/templates/extras/plugin_detail.html +3 -7
- nautobot/extras/templates/extras/plugins_list.html +0 -2
- nautobot/extras/tests/test_dynamicgroups.py +73 -18
- nautobot/extras/tests/test_views.py +5 -0
- nautobot/extras/urls.py +2 -94
- nautobot/extras/views.py +424 -430
- nautobot/ipam/querysets.py +3 -3
- nautobot/ipam/signals.py +6 -1
- nautobot/ipam/templates/ipam/prefix.html +0 -8
- nautobot/ipam/tests/test_api.py +5 -0
- nautobot/ipam/tests/test_models.py +387 -0
- nautobot/ipam/tests/test_querysets.py +46 -0
- nautobot/ipam/utils/migrations.py +1 -1
- nautobot/ipam/views.py +17 -8
- nautobot/project-static/docs/code-reference/nautobot/apps/testing.html +72 -0
- nautobot/project-static/docs/code-reference/nautobot/apps/ui.html +45 -9
- nautobot/project-static/docs/code-reference/nautobot/apps/views.html +393 -15
- nautobot/project-static/docs/development/apps/api/nautobot-app-config.html +1 -1
- nautobot/project-static/docs/development/core/getting-started.html +0 -15
- nautobot/project-static/docs/development/core/ui-component-framework.html +6 -11
- nautobot/project-static/docs/objects.inv +0 -0
- nautobot/project-static/docs/release-notes/version-2.4.html +222 -0
- nautobot/project-static/docs/search/search_index.json +1 -1
- nautobot/project-static/docs/sitemap.xml +300 -300
- nautobot/project-static/docs/sitemap.xml.gz +0 -0
- nautobot/project-static/docs/user-guide/administration/configuration/settings.html +27 -0
- nautobot/project-static/img/nautobot_icon.svg +32 -34
- nautobot/project-static/js/table_sorting_indicator.js +0 -2
- {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/METADATA +4 -4
- {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/RECORD +90 -85
- nautobot/core/templates/inc/breadcrumbs.html +0 -14
- nautobot/project-static/docs/requirements.txt +0 -14
- {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/LICENSE.txt +0 -0
- {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/NOTICE +0 -0
- {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/WHEEL +0 -0
- {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/entry_points.txt +0 -0
|
@@ -0,0 +1,199 @@
|
|
|
1
|
+
{% extends 'generic/object_create.html' %}
|
|
2
|
+
{% load static %}
|
|
3
|
+
{% load form_helpers %}
|
|
4
|
+
|
|
5
|
+
{% block form_errors %}
|
|
6
|
+
{% if form.non_field_errors %}
|
|
7
|
+
<div class="panel panel-danger">
|
|
8
|
+
<div class="panel-heading"><strong>Errors</strong></div>
|
|
9
|
+
<div class="panel-body">
|
|
10
|
+
{{ form.non_field_errors }}
|
|
11
|
+
{% for child in children.forms %}
|
|
12
|
+
{% if child.errors %}
|
|
13
|
+
{% for error in child.errors.values %}{{ error }}{% endfor %}
|
|
14
|
+
{% endif %}
|
|
15
|
+
{% endfor %}
|
|
16
|
+
</div>
|
|
17
|
+
</div>
|
|
18
|
+
{% endif %}
|
|
19
|
+
{% endblock %}
|
|
20
|
+
|
|
21
|
+
{% block form %}
|
|
22
|
+
<div class="panel panel-default">
|
|
23
|
+
<div class="panel-heading"><strong>Dynamic Group</strong></div>
|
|
24
|
+
<div class="panel-body">
|
|
25
|
+
{% render_field form.name %}
|
|
26
|
+
{% render_field form.description %}
|
|
27
|
+
{% render_field form.content_type %}
|
|
28
|
+
{% render_field form.group_type %}
|
|
29
|
+
{% render_field form.tenant %}
|
|
30
|
+
</div>
|
|
31
|
+
</div>
|
|
32
|
+
<div class="panel panel-default">
|
|
33
|
+
<div class="panel-heading"><strong>Filter Options</strong></div>
|
|
34
|
+
<div class="panel-body">
|
|
35
|
+
<ul class="nav nav-tabs" role="tablist">
|
|
36
|
+
<li role="presentation" class="active">
|
|
37
|
+
<a href="#filter-form" role="tab" data-toggle="tab">Filter Fields</a>
|
|
38
|
+
</li>
|
|
39
|
+
<li role="presentation">
|
|
40
|
+
<a href="#children-form" role="tab" data-toggle="tab">Child Groups</a>
|
|
41
|
+
</li>
|
|
42
|
+
</ul>
|
|
43
|
+
<div class="tab-content">
|
|
44
|
+
<div class="tab-pane active" id="filter-form">
|
|
45
|
+
{% if filter_form %}
|
|
46
|
+
<span class="help-block">
|
|
47
|
+
Select the filtering criteria to determine membership of objects matching
|
|
48
|
+
the Content Type for this Dynamic Group. Fields that are not a dropdown are
|
|
49
|
+
expected to have string inputs and do not support multiple values.
|
|
50
|
+
</span>
|
|
51
|
+
|
|
52
|
+
<div class="panel panel-default">
|
|
53
|
+
<div class="panel-heading"><strong>Object Fields</strong></div>
|
|
54
|
+
<div class="panel-body">
|
|
55
|
+
{% render_form filter_form excluded_fields="[]" %}
|
|
56
|
+
</div>
|
|
57
|
+
</div>
|
|
58
|
+
|
|
59
|
+
{% if filter_form.custom_fields %}
|
|
60
|
+
<div class="panel panel-default">
|
|
61
|
+
<div class="panel-heading"><strong>Custom Fields</strong></div>
|
|
62
|
+
<div class="panel-body">
|
|
63
|
+
{% render_custom_fields filter_form %}
|
|
64
|
+
</div>
|
|
65
|
+
</div>
|
|
66
|
+
{% endif %}
|
|
67
|
+
|
|
68
|
+
{% if filter_form.relationships %}
|
|
69
|
+
<div class="panel panel-default">
|
|
70
|
+
<div class="panel-heading"><strong>Relationships</strong></div>
|
|
71
|
+
<div class="panel-body">
|
|
72
|
+
{% render_relationships filter_form %}
|
|
73
|
+
</div>
|
|
74
|
+
</div>
|
|
75
|
+
{% endif %}
|
|
76
|
+
|
|
77
|
+
{% else %}
|
|
78
|
+
<span class="help-block">
|
|
79
|
+
Filtering criteria will be available after initially saving this group and
|
|
80
|
+
returning to this page.
|
|
81
|
+
</span>
|
|
82
|
+
{% endif %}
|
|
83
|
+
</div>
|
|
84
|
+
<div class="tab-pane" id="children-form">
|
|
85
|
+
{% if children.errors %}
|
|
86
|
+
<div class="text-danger">
|
|
87
|
+
Please correct the error(s) below:
|
|
88
|
+
|
|
89
|
+
{% for child in children.forms %}
|
|
90
|
+
{% if child.errors %}
|
|
91
|
+
{% for error in child.errors.values %}{{ error }}{% endfor %}
|
|
92
|
+
{% endif %}
|
|
93
|
+
{% endfor %}
|
|
94
|
+
</div>
|
|
95
|
+
{% endif %}
|
|
96
|
+
{{ children.non_field_errors }}
|
|
97
|
+
<table class="table" id="children">
|
|
98
|
+
{{ children.management_form }}
|
|
99
|
+
{% for child_form in children.forms %}
|
|
100
|
+
{% if forloop.first %}
|
|
101
|
+
<thead>
|
|
102
|
+
<tr>
|
|
103
|
+
<th>Weight</th>
|
|
104
|
+
{% for field in child_form.visible_fields %}
|
|
105
|
+
<th>{{ field.label|capfirst }}</th>
|
|
106
|
+
{% endfor %}
|
|
107
|
+
</tr>
|
|
108
|
+
</thead>
|
|
109
|
+
{% endif %}
|
|
110
|
+
<tr class="formset_row-{{ children.prefix }}">
|
|
111
|
+
<td><i class="mdi mdi-drag-horizontal drag-handler"></i></td>
|
|
112
|
+
{% for field in child_form.visible_fields %}
|
|
113
|
+
<td>
|
|
114
|
+
{% if forloop.first %}
|
|
115
|
+
{% for hidden in child_form.hidden_fields %}
|
|
116
|
+
{{ hidden }}
|
|
117
|
+
{% endfor %}
|
|
118
|
+
{% endif %}
|
|
119
|
+
{{ field }}
|
|
120
|
+
{% if field.errors %}
|
|
121
|
+
<ul>
|
|
122
|
+
{% for error in field.errors %}
|
|
123
|
+
{# Embed an HTML comment indicating the error for extraction by tests #}
|
|
124
|
+
<!-- FORM-ERROR {{ field.name }}: {{ error }} -->
|
|
125
|
+
<li class="text-danger">{{ error }}</li>
|
|
126
|
+
{% endfor %}
|
|
127
|
+
</ul>
|
|
128
|
+
{% endif %}
|
|
129
|
+
</td>
|
|
130
|
+
{% endfor %}
|
|
131
|
+
</tr>
|
|
132
|
+
{% endfor %}
|
|
133
|
+
</table>
|
|
134
|
+
</div>
|
|
135
|
+
</div>
|
|
136
|
+
</div>
|
|
137
|
+
</div>
|
|
138
|
+
{% include 'inc/extras_features_edit_form_fields.html' %}
|
|
139
|
+
{% endblock form %}
|
|
140
|
+
|
|
141
|
+
{% block javascript %}
|
|
142
|
+
{{ block.super }}
|
|
143
|
+
<script src="{% static 'jquery/jquery.formset.js' %}"></script>
|
|
144
|
+
<script type="text/javascript">
|
|
145
|
+
$('.formset_row-{{ children.prefix }}').formset({
|
|
146
|
+
addText: '<span class="mdi mdi-plus-thick" aria-hidden="true"></span> Add another Group',
|
|
147
|
+
addCssClass: 'btn btn-primary add-row',
|
|
148
|
+
deleteText: '<span class="mdi mdi-trash-can-outline" aria-hidden="true"></span>',
|
|
149
|
+
deleteCssClass: 'btn btn-danger delete-row',
|
|
150
|
+
prefix: '{{ children.prefix }}',
|
|
151
|
+
formCssClass: 'dynamic-formset-{{ children.prefix }}',
|
|
152
|
+
added: jsify_form
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
// Function that takes a Dynamic Group children form and recomputes the weights.
|
|
156
|
+
function update_weights(children_form){
|
|
157
|
+
let w = 10;
|
|
158
|
+
$(children_form).find('.formset_row-{{ children.prefix }}').each((i, el) => {
|
|
159
|
+
|
|
160
|
+
// A child is considered "weight-able" if any of the settable values contains a value
|
|
161
|
+
// If a user is in the middle of setting a group we want to consider it for weight
|
|
162
|
+
// We also want to remove the weight if it's empty to not throw a field value error on a hidden field
|
|
163
|
+
//
|
|
164
|
+
// el = child row
|
|
165
|
+
// sel = a select element either specifying the child group or operator
|
|
166
|
+
// we don't need to worry about which one we are looking at, just if it has a value
|
|
167
|
+
let contains_value = $(el).find("select[name$='-group'], select[name$='-operator']") // Get any select element in this row ending in -group or -operator
|
|
168
|
+
.toArray() // Convert to Array to access .some() method
|
|
169
|
+
.some((x) => { return ($(x).val().length > 0) === true }) // Evaluate if any element in this row has a value length over 0
|
|
170
|
+
// .some(fn(x)) will return a true/false if any item in the array return true when passed into the evaluator function
|
|
171
|
+
|
|
172
|
+
|
|
173
|
+
|
|
174
|
+
let weight_input = $(el).find("*[name$='-weight']")
|
|
175
|
+
weight_input.val("") // Reset the value to start clean always
|
|
176
|
+
if(contains_value) {
|
|
177
|
+
weight_input.val(w); // Set a weight
|
|
178
|
+
w += 10; // Following current convention by allowing space between children to aid in API interface.
|
|
179
|
+
// If someone wants to add a child later via the API they can insert a child between children without
|
|
180
|
+
// having to re-weight all children following first.
|
|
181
|
+
}
|
|
182
|
+
})
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
// Make the children form formset rows draggable and hook to re-weight on drag complete.
|
|
186
|
+
$('#children-form').sortable({
|
|
187
|
+
handle: '.drag-handler',
|
|
188
|
+
items: '.formset_row-{{ children.prefix }}',
|
|
189
|
+
update: ( e, ui ) => {
|
|
190
|
+
update_weights(e.target)
|
|
191
|
+
}
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
// Instead of re-weight on every input change, just do it before we submit the form.
|
|
195
|
+
$('form.form').submit((e) => {
|
|
196
|
+
$(e.target).find('#children-form').each((i, cf) => { update_weights(cf) });
|
|
197
|
+
});
|
|
198
|
+
</script>
|
|
199
|
+
{% endblock javascript %}
|
|
@@ -1,82 +1,2 @@
|
|
|
1
|
-
{% extends '
|
|
2
|
-
{%
|
|
3
|
-
|
|
4
|
-
{% block extra_buttons %}
|
|
5
|
-
{% if perms.extras.change_gitrepository %}
|
|
6
|
-
<form class="form-inline" style="display: inline-block"
|
|
7
|
-
method="post" action="{% url 'extras:gitrepository_dryrun' pk=object.pk %}">
|
|
8
|
-
{% csrf_token %}
|
|
9
|
-
<button type="submit" class="btn btn-info">
|
|
10
|
-
<i class="mdi mdi-book-refresh" aria-hidden="true"></i> Dry-Run
|
|
11
|
-
</button>
|
|
12
|
-
</form>
|
|
13
|
-
<form class="form-inline" style="display: inline-block"
|
|
14
|
-
method="post" action="{% url 'extras:gitrepository_sync' pk=object.pk %}">
|
|
15
|
-
{% csrf_token %}
|
|
16
|
-
<button type="submit" class="btn btn-primary">
|
|
17
|
-
<i class="mdi mdi-source-branch-sync" aria-hidden="true"></i> Sync
|
|
18
|
-
</button>
|
|
19
|
-
</form>
|
|
20
|
-
{% endif %}
|
|
21
|
-
{% endblock extra_buttons %}
|
|
22
|
-
|
|
23
|
-
{% block extra_nav_tabs %}
|
|
24
|
-
<li role="presentation"{% if active_tab == 'result' %} class="active"{% endif %}>
|
|
25
|
-
<a href="{% url 'extras:gitrepository_result' pk=object.pk %}">Synchronization Status</a>
|
|
26
|
-
</li>
|
|
27
|
-
{% endblock extra_nav_tabs %}
|
|
28
|
-
|
|
29
|
-
{% block content_left_page %}
|
|
30
|
-
<div class="panel panel-default">
|
|
31
|
-
<div class="panel-heading">
|
|
32
|
-
<strong>Repository Details</strong>
|
|
33
|
-
</div>
|
|
34
|
-
<table class="table table-hover panel-body attr-table">
|
|
35
|
-
<tr>
|
|
36
|
-
<td>Remote URL</td>
|
|
37
|
-
<td>{{ object.remote_url }}</td>
|
|
38
|
-
</tr>
|
|
39
|
-
<tr>
|
|
40
|
-
<td>Branch</td>
|
|
41
|
-
<td>
|
|
42
|
-
<code>{{ object.branch }}</code>
|
|
43
|
-
{% if object.current_head %}
|
|
44
|
-
(checked out locally at commit <code>{{ object.current_head }}</code>)
|
|
45
|
-
{% else %}
|
|
46
|
-
(not locally checked out yet)
|
|
47
|
-
{% endif %}
|
|
48
|
-
</td>
|
|
49
|
-
</tr>
|
|
50
|
-
<tr>
|
|
51
|
-
<td>Secrets Group</td>
|
|
52
|
-
<td>{{ object.secrets_group|hyperlinked_object }}</td>
|
|
53
|
-
</tr>
|
|
54
|
-
</table>
|
|
55
|
-
</div>
|
|
56
|
-
{% endblock content_left_page %}
|
|
57
|
-
|
|
58
|
-
{% block content_right_page %}
|
|
59
|
-
<div class="panel panel-default">
|
|
60
|
-
<div class="panel-heading">
|
|
61
|
-
<strong>Provided Data Types</strong>
|
|
62
|
-
</div>
|
|
63
|
-
<table class="table table-hover panel-body">
|
|
64
|
-
{% for entry in datasource_contents %}
|
|
65
|
-
<tr>
|
|
66
|
-
<td>
|
|
67
|
-
<span style="display: inline-block" class="label label-info">
|
|
68
|
-
<i class="mdi {{ entry.icon }}"></i>
|
|
69
|
-
</span>
|
|
70
|
-
{{ entry.name|title }}</td>
|
|
71
|
-
<td>
|
|
72
|
-
{% if entry.content_identifier in object.provided_contents %}
|
|
73
|
-
{{ True | render_boolean }}
|
|
74
|
-
{% else %}
|
|
75
|
-
{{ False | render_boolean }}
|
|
76
|
-
{% endif %}
|
|
77
|
-
</td>
|
|
78
|
-
</tr>
|
|
79
|
-
{% endfor %}
|
|
80
|
-
</table>
|
|
81
|
-
</div>
|
|
82
|
-
{% endblock %}
|
|
1
|
+
{% extends 'extras/gitrepository_retrieve.html' %}
|
|
2
|
+
{% comment %}3.0 TODO: remove this template, which only exists for backward compatibility with 2.4 and earlier{% endcomment %}
|
|
@@ -1,13 +1,2 @@
|
|
|
1
|
-
{% extends '
|
|
2
|
-
|
|
3
|
-
{% block buttons %}
|
|
4
|
-
{% if editing %}
|
|
5
|
-
<button type="submit" name="_dryrun_update" class="btn btn-warning">Update & Dry Run</button>
|
|
6
|
-
<button type="submit" name="_update" class="btn btn-primary">Update & Sync</button>
|
|
7
|
-
{% else %}
|
|
8
|
-
<button type="submit" name="_dryrun_create" class="btn btn-info">Create & Dry Run</button>
|
|
9
|
-
<button type="submit" name="_create" class="btn btn-primary">Create & Sync</button>
|
|
10
|
-
<button type="submit" name="_addanother" class="btn btn-primary">Create and Add Another</button>
|
|
11
|
-
{% endif %}
|
|
12
|
-
<a href="{{ return_url }}" class="btn btn-default">Cancel</a>
|
|
13
|
-
{% endblock %}
|
|
1
|
+
{% extends 'extras/gitrepository_update.html' %}
|
|
2
|
+
{% comment %}3.0 TODO: remove this template, which only exists for backward compatibility with 2.4 and earlier{% endcomment %}
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
{% extends 'generic/object_retrieve.html' %}
|
|
2
|
+
{% load helpers %}
|
|
3
|
+
|
|
4
|
+
{% block extra_buttons %}
|
|
5
|
+
{% if perms.extras.change_gitrepository %}
|
|
6
|
+
<form class="form-inline" style="display: inline-block"
|
|
7
|
+
method="post" action="{% url 'extras:gitrepository_dryrun' pk=object.pk %}">
|
|
8
|
+
{% csrf_token %}
|
|
9
|
+
<button type="submit" class="btn btn-info">
|
|
10
|
+
<i class="mdi mdi-book-refresh" aria-hidden="true"></i> Dry-Run
|
|
11
|
+
</button>
|
|
12
|
+
</form>
|
|
13
|
+
<form class="form-inline" style="display: inline-block"
|
|
14
|
+
method="post" action="{% url 'extras:gitrepository_sync' pk=object.pk %}">
|
|
15
|
+
{% csrf_token %}
|
|
16
|
+
<button type="submit" class="btn btn-primary">
|
|
17
|
+
<i class="mdi mdi-source-branch-sync" aria-hidden="true"></i> Sync
|
|
18
|
+
</button>
|
|
19
|
+
</form>
|
|
20
|
+
{% endif %}
|
|
21
|
+
{% endblock extra_buttons %}
|
|
22
|
+
|
|
23
|
+
{% block extra_nav_tabs %}
|
|
24
|
+
<li role="presentation"{% if active_tab == 'result' %} class="active"{% endif %}>
|
|
25
|
+
<a href="{% url 'extras:gitrepository_result' pk=object.pk %}">Synchronization Status</a>
|
|
26
|
+
</li>
|
|
27
|
+
{% endblock extra_nav_tabs %}
|
|
28
|
+
|
|
29
|
+
{% block content_left_page %}
|
|
30
|
+
<div class="panel panel-default">
|
|
31
|
+
<div class="panel-heading">
|
|
32
|
+
<strong>Repository Details</strong>
|
|
33
|
+
</div>
|
|
34
|
+
<table class="table table-hover panel-body attr-table">
|
|
35
|
+
<tr>
|
|
36
|
+
<td>Remote URL</td>
|
|
37
|
+
<td>{{ object.remote_url }}</td>
|
|
38
|
+
</tr>
|
|
39
|
+
<tr>
|
|
40
|
+
<td>Branch</td>
|
|
41
|
+
<td>
|
|
42
|
+
<code>{{ object.branch }}</code>
|
|
43
|
+
{% if object.current_head %}
|
|
44
|
+
(checked out locally at commit <code>{{ object.current_head }}</code>)
|
|
45
|
+
{% else %}
|
|
46
|
+
(not locally checked out yet)
|
|
47
|
+
{% endif %}
|
|
48
|
+
</td>
|
|
49
|
+
</tr>
|
|
50
|
+
<tr>
|
|
51
|
+
<td>Secrets Group</td>
|
|
52
|
+
<td>{{ object.secrets_group|hyperlinked_object }}</td>
|
|
53
|
+
</tr>
|
|
54
|
+
</table>
|
|
55
|
+
</div>
|
|
56
|
+
{% endblock content_left_page %}
|
|
57
|
+
|
|
58
|
+
{% block content_right_page %}
|
|
59
|
+
<div class="panel panel-default">
|
|
60
|
+
<div class="panel-heading">
|
|
61
|
+
<strong>Provided Data Types</strong>
|
|
62
|
+
</div>
|
|
63
|
+
<table class="table table-hover panel-body">
|
|
64
|
+
{% for entry in datasource_contents %}
|
|
65
|
+
<tr>
|
|
66
|
+
<td>
|
|
67
|
+
<span style="display: inline-block" class="label label-info">
|
|
68
|
+
<i class="mdi {{ entry.icon }}"></i>
|
|
69
|
+
</span>
|
|
70
|
+
{{ entry.name|title }}</td>
|
|
71
|
+
<td>
|
|
72
|
+
{% if entry.content_identifier in object.provided_contents %}
|
|
73
|
+
{{ True | render_boolean }}
|
|
74
|
+
{% else %}
|
|
75
|
+
{{ False | render_boolean }}
|
|
76
|
+
{% endif %}
|
|
77
|
+
</td>
|
|
78
|
+
</tr>
|
|
79
|
+
{% endfor %}
|
|
80
|
+
</table>
|
|
81
|
+
</div>
|
|
82
|
+
{% endblock %}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
{% extends 'generic/object_create.html' %}
|
|
2
|
+
|
|
3
|
+
{% block buttons %}
|
|
4
|
+
{% if editing %}
|
|
5
|
+
<button type="submit" name="_dryrun_update" class="btn btn-warning">Update & Dry Run</button>
|
|
6
|
+
<button type="submit" name="_update" class="btn btn-primary">Update & Sync</button>
|
|
7
|
+
{% else %}
|
|
8
|
+
<button type="submit" name="_dryrun_create" class="btn btn-info">Create & Dry Run</button>
|
|
9
|
+
<button type="submit" name="_create" class="btn btn-primary">Create & Sync</button>
|
|
10
|
+
<button type="submit" name="_addanother" class="btn btn-primary">Create and Add Another</button>
|
|
11
|
+
{% endif %}
|
|
12
|
+
<a href="{{ return_url }}" class="btn btn-default">Cancel</a>
|
|
13
|
+
{% endblock %}
|
|
@@ -1,53 +1 @@
|
|
|
1
1
|
{% extends 'generic/object_retrieve.html' %}
|
|
2
|
-
{% load buttons %}
|
|
3
|
-
{% load plugins %}
|
|
4
|
-
{% load perms %}
|
|
5
|
-
{% load helpers %}
|
|
6
|
-
|
|
7
|
-
{% block breadcrumbs %}
|
|
8
|
-
<li><a href="{% url 'extras:note_list' %}">Notes</a></li>
|
|
9
|
-
{% if object.assigned_object.get_absolute_url %}
|
|
10
|
-
<li><a href="{{ object.assigned_object.get_absolute_url }}notes/">{{ object.assigned_object }}</a></li>
|
|
11
|
-
{% endif %}
|
|
12
|
-
<li>{{ object }}</li>
|
|
13
|
-
{% endblock breadcrumbs %}
|
|
14
|
-
|
|
15
|
-
{% block content %}
|
|
16
|
-
<div class="row">
|
|
17
|
-
<div class="col-md-5">
|
|
18
|
-
<div class="panel panel-default">
|
|
19
|
-
<div class="panel-heading">
|
|
20
|
-
<strong>Note</strong>
|
|
21
|
-
</div>
|
|
22
|
-
<table class="table table-hover panel-body attr-table">
|
|
23
|
-
<tr>
|
|
24
|
-
<td>User</td>
|
|
25
|
-
<td>
|
|
26
|
-
{{ object.user|default:object.user_name }}
|
|
27
|
-
</td>
|
|
28
|
-
</tr>
|
|
29
|
-
<tr>
|
|
30
|
-
<td>Object Type</td>
|
|
31
|
-
<td>
|
|
32
|
-
{{ object.assigned_object_type }}
|
|
33
|
-
</td>
|
|
34
|
-
</tr>
|
|
35
|
-
<tr>
|
|
36
|
-
<td>Object</td>
|
|
37
|
-
<td>
|
|
38
|
-
{{ object.assigned_object | hyperlinked_object }}
|
|
39
|
-
</td>
|
|
40
|
-
</tr>
|
|
41
|
-
</table>
|
|
42
|
-
</div>
|
|
43
|
-
<div class="panel panel-default">
|
|
44
|
-
<div class="panel-heading">
|
|
45
|
-
<strong>Text</strong>
|
|
46
|
-
</div>
|
|
47
|
-
<div class="panel-body rendered-markdown">
|
|
48
|
-
{{ object.note|render_markdown }}
|
|
49
|
-
</div>
|
|
50
|
-
</div>
|
|
51
|
-
</div>
|
|
52
|
-
</div>
|
|
53
|
-
{% endblock %}
|
|
@@ -1,15 +1,11 @@
|
|
|
1
1
|
{% extends 'generic/object_retrieve.html' %}
|
|
2
2
|
{% load helpers %}
|
|
3
|
+
{% load ui_framework %}
|
|
3
4
|
|
|
4
5
|
{% block header %}
|
|
5
6
|
<div class="row noprint">
|
|
6
7
|
<div class="col-md-12">
|
|
7
|
-
|
|
8
|
-
{% block breadcrumbs %}
|
|
9
|
-
<li><a href="{% url 'apps:apps_list' %}">Installed Apps</a></li>
|
|
10
|
-
<li><a href="{% url 'apps:app_detail' app=app_data.package %}">{{ app_data.name | bettertitle }}</a></li>
|
|
11
|
-
{% endblock breadcrumbs %}
|
|
12
|
-
</ol>
|
|
8
|
+
{% render_breadcrumbs %}
|
|
13
9
|
</div>
|
|
14
10
|
</div>
|
|
15
11
|
<div class="pull-right noprint">
|
|
@@ -32,7 +28,7 @@
|
|
|
32
28
|
{% endblock buttons %}
|
|
33
29
|
</div>
|
|
34
30
|
{% block masthead %}
|
|
35
|
-
<h1>{% block title %}{
|
|
31
|
+
<h1>{% block title %}{% render_title %}{% endblock %}</h1>
|
|
36
32
|
{% endblock masthead %}
|
|
37
33
|
{% endblock header %}
|
|
38
34
|
|
|
@@ -18,6 +18,7 @@ from nautobot.dcim.models import (
|
|
|
18
18
|
Device,
|
|
19
19
|
DeviceType,
|
|
20
20
|
FrontPort,
|
|
21
|
+
InventoryItem,
|
|
21
22
|
Location,
|
|
22
23
|
LocationType,
|
|
23
24
|
Manufacturer,
|
|
@@ -107,6 +108,24 @@ class DynamicGroupTestBase(TestCase):
|
|
|
107
108
|
),
|
|
108
109
|
]
|
|
109
110
|
|
|
111
|
+
cls.inventory_items = [
|
|
112
|
+
InventoryItem.objects.create(
|
|
113
|
+
name="device-location-1-item-1",
|
|
114
|
+
serial="location-1-item-1",
|
|
115
|
+
device=cls.devices[0],
|
|
116
|
+
),
|
|
117
|
+
InventoryItem.objects.create(
|
|
118
|
+
name="device-location-1-item-2",
|
|
119
|
+
serial="location-1-item-2",
|
|
120
|
+
device=cls.devices[0],
|
|
121
|
+
),
|
|
122
|
+
InventoryItem.objects.create(
|
|
123
|
+
name="device-location-2-item-1",
|
|
124
|
+
serial="location-2-item-1",
|
|
125
|
+
device=cls.devices[1],
|
|
126
|
+
),
|
|
127
|
+
]
|
|
128
|
+
|
|
110
129
|
cls.groups = [
|
|
111
130
|
DynamicGroup.objects.create(
|
|
112
131
|
name="Parent",
|
|
@@ -152,7 +171,13 @@ class DynamicGroupTestBase(TestCase):
|
|
|
152
171
|
DynamicGroup.objects.create(
|
|
153
172
|
name="MultiValueCharFilter",
|
|
154
173
|
description="A group with a multivaluechar filter",
|
|
155
|
-
filter={"name": ["device-1", "device-2", "device-3"]},
|
|
174
|
+
filter={"name": ["device-location-1", "device-location-2", "device-location-3"]},
|
|
175
|
+
content_type=cls.device_ct,
|
|
176
|
+
),
|
|
177
|
+
DynamicGroup.objects.create(
|
|
178
|
+
name="SearchFilter",
|
|
179
|
+
description="A group with a search filter",
|
|
180
|
+
filter={"q": "location-1"},
|
|
156
181
|
content_type=cls.device_ct,
|
|
157
182
|
),
|
|
158
183
|
]
|
|
@@ -165,6 +190,8 @@ class DynamicGroupTestBase(TestCase):
|
|
|
165
190
|
cls.third_child = cls.groups[3]
|
|
166
191
|
cls.nested_child = cls.groups[4]
|
|
167
192
|
cls.no_match_filter = cls.groups[5]
|
|
193
|
+
cls.multivaluechar_filter = cls.groups[6]
|
|
194
|
+
cls.search_filter = cls.groups[7]
|
|
168
195
|
cls.invalid_filter = DynamicGroup.objects.create(
|
|
169
196
|
name="Invalid Filter",
|
|
170
197
|
description="A group with a filter that's invalid",
|
|
@@ -289,6 +316,9 @@ class DynamicGroupModelTest(DynamicGroupTestBase): # TODO: BaseModelTestCase mi
|
|
|
289
316
|
self.assertIn(device1, group.members)
|
|
290
317
|
self.assertNotIn(device2, group.members)
|
|
291
318
|
|
|
319
|
+
group = self.search_filter
|
|
320
|
+
self.assertIn(device1, group.members)
|
|
321
|
+
|
|
292
322
|
def test_static_member_operations(self):
|
|
293
323
|
sg = DynamicGroup.objects.create(
|
|
294
324
|
name="All Prefixes",
|
|
@@ -437,17 +467,24 @@ class DynamicGroupModelTest(DynamicGroupTestBase): # TODO: BaseModelTestCase mi
|
|
|
437
467
|
|
|
438
468
|
def test_count(self):
|
|
439
469
|
"""Test `DynamicGroup.count`."""
|
|
440
|
-
expected =
|
|
441
|
-
self.parent
|
|
442
|
-
self.first_child
|
|
443
|
-
self.second_child
|
|
444
|
-
self.third_child
|
|
445
|
-
self.nested_child
|
|
446
|
-
self.no_match_filter
|
|
447
|
-
self.
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
|
|
470
|
+
expected = [
|
|
471
|
+
(self.parent, 1),
|
|
472
|
+
(self.first_child, 1),
|
|
473
|
+
(self.second_child, 1),
|
|
474
|
+
(self.third_child, 2),
|
|
475
|
+
(self.nested_child, 2),
|
|
476
|
+
(self.no_match_filter, 0),
|
|
477
|
+
(self.multivaluechar_filter, 3),
|
|
478
|
+
(self.search_filter, 1),
|
|
479
|
+
(self.invalid_filter, 0),
|
|
480
|
+
]
|
|
481
|
+
for grp, cnt in expected:
|
|
482
|
+
with self.subTest(grp.name):
|
|
483
|
+
self.assertEqual(grp.count, cnt, list(grp.members.values_list("name", flat=True)))
|
|
484
|
+
self.assertEqual(grp.members.count(), cnt, list(grp.members.values_list("name", flat=True)))
|
|
485
|
+
# https://github.com/nautobot/nautobot/issues/7631
|
|
486
|
+
members = grp.update_cached_members()
|
|
487
|
+
self.assertEqual(members.count(), cnt, list(members.values_list("name", flat=True)))
|
|
451
488
|
|
|
452
489
|
def test_model(self):
|
|
453
490
|
"""Test `DynamicGroup.model`."""
|
|
@@ -635,9 +672,9 @@ class DynamicGroupModelTest(DynamicGroupTestBase): # TODO: BaseModelTestCase mi
|
|
|
635
672
|
group1 = self.first_child # Filter has `location`
|
|
636
673
|
self.assertEqual(group1.get_initial(), group1.filter)
|
|
637
674
|
# Test if MultiValueCharField is properly pre-populated
|
|
638
|
-
group2 = self.
|
|
675
|
+
group2 = self.multivaluechar_filter # Filter has `name`
|
|
639
676
|
initial = group2.get_initial()
|
|
640
|
-
expected = {"name": ["device-1", "device-2", "device-3"]}
|
|
677
|
+
expected = {"name": ["device-location-1", "device-location-2", "device-location-3"]}
|
|
641
678
|
self.assertEqual(initial, expected)
|
|
642
679
|
|
|
643
680
|
def test_set_filter(self):
|
|
@@ -1152,7 +1189,11 @@ class DynamicGroupMixinModelTest(DynamicGroupTestBase):
|
|
|
1152
1189
|
with self.assertNumQueries(1):
|
|
1153
1190
|
qs = self.devices[0].dynamic_groups
|
|
1154
1191
|
list(qs)
|
|
1155
|
-
self.assertQuerysetEqualAndNotEmpty(
|
|
1192
|
+
self.assertQuerysetEqualAndNotEmpty(
|
|
1193
|
+
qs,
|
|
1194
|
+
[self.first_child, self.third_child, self.nested_child, self.multivaluechar_filter, self.search_filter],
|
|
1195
|
+
ordered=False,
|
|
1196
|
+
)
|
|
1156
1197
|
|
|
1157
1198
|
def test_dynamic_groups_cached(self):
|
|
1158
1199
|
for group in self.groups:
|
|
@@ -1160,21 +1201,35 @@ class DynamicGroupMixinModelTest(DynamicGroupTestBase):
|
|
|
1160
1201
|
with self.assertNumQueries(1):
|
|
1161
1202
|
qs = self.devices[0].dynamic_groups_cached
|
|
1162
1203
|
list(qs)
|
|
1163
|
-
self.assertQuerysetEqualAndNotEmpty(
|
|
1204
|
+
self.assertQuerysetEqualAndNotEmpty(
|
|
1205
|
+
qs,
|
|
1206
|
+
[self.first_child, self.third_child, self.nested_child, self.multivaluechar_filter, self.search_filter],
|
|
1207
|
+
ordered=False,
|
|
1208
|
+
)
|
|
1164
1209
|
|
|
1165
1210
|
def test_dynamic_groups_list(self):
|
|
1166
1211
|
for group in self.groups:
|
|
1167
1212
|
group.update_cached_members()
|
|
1168
1213
|
with self.assertNumQueries(1):
|
|
1169
1214
|
groups = self.devices[0].dynamic_groups_list
|
|
1170
|
-
self.assertEqual(
|
|
1215
|
+
self.assertEqual(
|
|
1216
|
+
set(groups),
|
|
1217
|
+
set(
|
|
1218
|
+
[self.first_child, self.third_child, self.nested_child, self.multivaluechar_filter, self.search_filter]
|
|
1219
|
+
),
|
|
1220
|
+
)
|
|
1171
1221
|
|
|
1172
1222
|
def test_dynamic_groups_list_cached(self):
|
|
1173
1223
|
for group in self.groups:
|
|
1174
1224
|
group.update_cached_members()
|
|
1175
1225
|
with self.assertNumQueries(1):
|
|
1176
1226
|
groups = self.devices[0].dynamic_groups_list_cached
|
|
1177
|
-
self.assertEqual(
|
|
1227
|
+
self.assertEqual(
|
|
1228
|
+
set(groups),
|
|
1229
|
+
set(
|
|
1230
|
+
[self.first_child, self.third_child, self.nested_child, self.multivaluechar_filter, self.search_filter]
|
|
1231
|
+
),
|
|
1232
|
+
)
|
|
1178
1233
|
|
|
1179
1234
|
|
|
1180
1235
|
class DynamicGroupFilterTest(DynamicGroupTestBase, FilterTestCases.FilterTestCase):
|