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.

Files changed (92) hide show
  1. nautobot/apps/views.py +2 -0
  2. nautobot/circuits/templates/circuits/circuittermination_retrieve.html +1 -8
  3. nautobot/circuits/templates/circuits/inc/circuit_termination_speed_fragment.html +9 -0
  4. nautobot/circuits/tests/integration/test_circuit.py +2 -2
  5. nautobot/circuits/views.py +32 -15
  6. nautobot/core/filters.py +2 -2
  7. nautobot/core/settings.py +1 -0
  8. nautobot/core/settings.yaml +9 -0
  9. nautobot/core/tables.py +21 -23
  10. nautobot/core/templates/components/breadcrumbs.html +19 -0
  11. nautobot/core/templates/generic/object_changelog.html +0 -2
  12. nautobot/core/templates/generic/object_list.html +15 -12
  13. nautobot/core/templates/generic/object_notes.html +0 -2
  14. nautobot/core/templates/generic/object_retrieve.html +16 -9
  15. nautobot/core/templatetags/helpers.py +24 -0
  16. nautobot/core/templatetags/ui_framework.py +40 -5
  17. nautobot/core/testing/filters.py +37 -21
  18. nautobot/core/testing/views.py +25 -0
  19. nautobot/core/tests/test_tables.py +43 -6
  20. nautobot/core/tests/test_templatetags_ui_framework.py +146 -0
  21. nautobot/core/tests/test_titles.py +2 -2
  22. nautobot/core/tests/test_ui.py +14 -1
  23. nautobot/core/tests/test_views.py +45 -0
  24. nautobot/core/ui/breadcrumbs.py +13 -8
  25. nautobot/core/ui/object_detail.py +43 -5
  26. nautobot/core/ui/titles.py +9 -5
  27. nautobot/core/views/__init__.py +24 -3
  28. nautobot/core/views/generic.py +42 -17
  29. nautobot/core/views/mixins.py +146 -12
  30. nautobot/core/views/utils.py +117 -0
  31. nautobot/dcim/models/devices.py +4 -0
  32. nautobot/dcim/tables/__init__.py +2 -0
  33. nautobot/dcim/tables/devices.py +24 -0
  34. nautobot/dcim/tables/power.py +2 -2
  35. nautobot/dcim/templates/dcim/device/base.html +1 -11
  36. nautobot/dcim/templates/dcim/device_component.html +0 -19
  37. nautobot/dcim/templates/dcim/modulebay_retrieve.html +0 -16
  38. nautobot/dcim/templates/dcim/virtualchassis_retrieve.html +1 -50
  39. nautobot/dcim/tests/test_views.py +41 -0
  40. nautobot/dcim/views.py +160 -39
  41. nautobot/extras/filters/mixins.py +1 -1
  42. nautobot/extras/forms/forms.py +15 -0
  43. nautobot/extras/models/groups.py +10 -1
  44. nautobot/extras/models/jobs.py +2 -2
  45. nautobot/extras/plugins/views.py +18 -5
  46. nautobot/extras/tables.py +4 -2
  47. nautobot/extras/templates/extras/customfield_retrieve.html +1 -128
  48. nautobot/extras/templates/extras/dynamicgroup.html +2 -99
  49. nautobot/extras/templates/extras/dynamicgroup_edit.html +2 -199
  50. nautobot/extras/templates/extras/dynamicgroup_retrieve.html +99 -0
  51. nautobot/extras/templates/extras/dynamicgroup_update.html +199 -0
  52. nautobot/extras/templates/extras/gitrepository.html +2 -82
  53. nautobot/extras/templates/extras/gitrepository_object_edit.html +2 -13
  54. nautobot/extras/templates/extras/gitrepository_retrieve.html +82 -0
  55. nautobot/extras/templates/extras/gitrepository_update.html +13 -0
  56. nautobot/extras/templates/extras/note_retrieve.html +0 -52
  57. nautobot/extras/templates/extras/plugin_detail.html +3 -7
  58. nautobot/extras/templates/extras/plugins_list.html +0 -2
  59. nautobot/extras/tests/test_dynamicgroups.py +73 -18
  60. nautobot/extras/tests/test_views.py +5 -0
  61. nautobot/extras/urls.py +2 -94
  62. nautobot/extras/views.py +424 -430
  63. nautobot/ipam/querysets.py +3 -3
  64. nautobot/ipam/signals.py +6 -1
  65. nautobot/ipam/templates/ipam/prefix.html +0 -8
  66. nautobot/ipam/tests/test_api.py +5 -0
  67. nautobot/ipam/tests/test_models.py +387 -0
  68. nautobot/ipam/tests/test_querysets.py +46 -0
  69. nautobot/ipam/utils/migrations.py +1 -1
  70. nautobot/ipam/views.py +17 -8
  71. nautobot/project-static/docs/code-reference/nautobot/apps/testing.html +72 -0
  72. nautobot/project-static/docs/code-reference/nautobot/apps/ui.html +45 -9
  73. nautobot/project-static/docs/code-reference/nautobot/apps/views.html +393 -15
  74. nautobot/project-static/docs/development/apps/api/nautobot-app-config.html +1 -1
  75. nautobot/project-static/docs/development/core/getting-started.html +0 -15
  76. nautobot/project-static/docs/development/core/ui-component-framework.html +6 -11
  77. nautobot/project-static/docs/objects.inv +0 -0
  78. nautobot/project-static/docs/release-notes/version-2.4.html +222 -0
  79. nautobot/project-static/docs/search/search_index.json +1 -1
  80. nautobot/project-static/docs/sitemap.xml +300 -300
  81. nautobot/project-static/docs/sitemap.xml.gz +0 -0
  82. nautobot/project-static/docs/user-guide/administration/configuration/settings.html +27 -0
  83. nautobot/project-static/img/nautobot_icon.svg +32 -34
  84. nautobot/project-static/js/table_sorting_indicator.js +0 -2
  85. {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/METADATA +4 -4
  86. {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/RECORD +90 -85
  87. nautobot/core/templates/inc/breadcrumbs.html +0 -14
  88. nautobot/project-static/docs/requirements.txt +0 -14
  89. {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/LICENSE.txt +0 -0
  90. {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/NOTICE +0 -0
  91. {nautobot-2.4.17.dist-info → nautobot-2.4.18.dist-info}/WHEEL +0 -0
  92. {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 '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 %}
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 '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
+ {% 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
- <ol class="breadcrumb">
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 %}{{ app_data.name | bettertitle }}{% endblock %}</h1>
31
+ <h1>{% block title %}{% render_title %}{% endblock %}</h1>
36
32
  {% endblock masthead %}
37
33
  {% endblock header %}
38
34
 
@@ -3,8 +3,6 @@
3
3
  {% load static %}
4
4
  {% load ui_framework %}
5
5
 
6
- {% block document_title %}{% render_title "plain" %}{% endblock %}
7
-
8
6
  {% block content %}
9
7
  <div class="row noprint">
10
8
  <div class="col-md-12">
@@ -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.count: 2,
442
- self.first_child.count: 1,
443
- self.second_child.count: 1,
444
- self.third_child.count: 2,
445
- self.nested_child.count: 2,
446
- self.no_match_filter.count: 0,
447
- self.invalid_filter.count: 0,
448
- }
449
- for grp, cnt in expected.items():
450
- self.assertEqual(grp, cnt)
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.groups[6] # Filter has `name`
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(qs, [self.first_child, self.third_child, self.nested_child], ordered=False)
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(qs, [self.first_child, self.third_child, self.nested_child], ordered=False)
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(set(groups), set([self.first_child, self.third_child, self.nested_child]))
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(set(groups), set([self.first_child, self.third_child, self.nested_child]))
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):