diff --git a/.github/ISSUE_TEMPLATE/dev-issue-template.md b/.github/ISSUE_TEMPLATE/dev-issue-template.md
index 530a6db49..8857901a5 100644
--- a/.github/ISSUE_TEMPLATE/dev-issue-template.md
+++ b/.github/ISSUE_TEMPLATE/dev-issue-template.md
@@ -9,6 +9,7 @@ assignees: ''
**Description:**
_Provide a brief background and justification for this issue_
+_OFA & UX team: Please include bullets capturing the value to STT end users, related guidance, and/or talking points that should be communicated to STTs in release notes_
**Acceptance Criteria:**
diff --git a/Taskfile.yml b/Taskfile.yml
index fa152a53f..8f1731fe9 100644
--- a/Taskfile.yml
+++ b/Taskfile.yml
@@ -2,6 +2,11 @@ version: '3'
tasks:
+ upload-kibana-objs:
+ desc: Upload dashboards to Kibana server
+ cmds:
+ - curl -X POST localhost:5601/api/saved_objects/_import -H "kbn-xsrf: true" --form file=@tdrs-backend/tdpservice/search_indexes/kibana_saved_objs.ndjson
+
create-network:
desc: Create the external network
cmds:
@@ -29,7 +34,7 @@ tasks:
desc: Create Sentry service
dir: sentry
cmds:
- # limiting the memory to 2GB and CPU to only one cpu @0, for faster response, you can remove the limittask : --cpuset-cpus 0
+ # limiting the memory to 2GB and CPU to only one cpu @0, for faster response, you can remove the limittask : --cpuset-cpus 0
- (docker run --privileged -p 9001:9000 -d --memory="8g" --memory-swap="8g" --name sentry docker:dind) || true
- docker exec sentry sh -c "git clone https://github.com/getsentry/self-hosted.git || true"
@@ -95,7 +100,7 @@ tasks:
desc: Restart backend web server
dir: tdrs-backend
cmds:
- - docker-compose -f docker-compose.yml restart -d
+ - docker-compose -f docker-compose.yml restart
backend-bash:
desc: Open a shell in the backend container
@@ -172,7 +177,7 @@ tasks:
- docker rm $(docker ps -aq) || true
- docker rmi $(docker images -q) || true
- docker volume rm $(docker volume ls -q) || true
-
+
clamav-up:
desc: Start clamav service
dir: tdrs-backend
@@ -195,7 +200,7 @@ tasks:
desc: Restart frontend web server
dir: tdrs-frontend
cmds:
- - docker-compose -f docker-compose.yml restart -d
+ - docker-compose -f docker-compose.yml restart
frontend-av:
desc: Start frontend with optional clamav service
@@ -204,7 +209,7 @@ tasks:
- task: frontend-up
- task: clamav-up
-
+
# need more work
frontend-init:
desc: Initialize the frontend project
diff --git a/docs/How-We-Work/team-meetings.md b/docs/How-We-Work/team-meetings.md
index 8b78dcc59..ff770ed7a 100644
--- a/docs/How-We-Work/team-meetings.md
+++ b/docs/How-We-Work/team-meetings.md
@@ -68,12 +68,12 @@ A typical sprint schedule is described in the table below.
- External factors outside of the author spending time building the ticket (ie need external team's input, see how a feature develops, etc.)
- Ex. Waiting on X ticket to finish the scope of said unrefined ticket, problem found / unsure how big it is and knows other work will unearth it
- If we know the ACs but not the tasks, then its unrefined
+ - Release Notes summary is empty or incomplete as applicable, to be provided by UX/Product by default.
- Refined: Ticket is complete and is ready to be executed.
- Refined & Ready to Go (Next Sprint)
- "Earmarked" work for the upcoming sprint.
- **Labelling:**
- WIP
- - Author knows the 5 W's or darn near (90%)
- Drafted ticket – either still on the author the finish their part or a short team conversation is needed.
- Administrative in nature
- Ex. Stub, ticket that doesn't feel there's enough to warrant an introduction
diff --git a/docs/Sprint-Review/sprint-106-summary.md b/docs/Sprint-Review/sprint-106-summary.md
new file mode 100644
index 000000000..6d211ed81
--- /dev/null
+++ b/docs/Sprint-Review/sprint-106-summary.md
@@ -0,0 +1,97 @@
+# sprint-106-summary
+
+8/14/202 - 8/27/2024
+
+### Priority Setting
+
+* Reparsing
+ * Reasoning: OFA has ability to reparse and the data process with latest system logic which will enable STTs to get error reports that are meaningful and relevant and OFA gets the needed data. System, as a whole, will be more flexible with data flow (able to hot fix and hit reparse). Additionally, we want to avoid missing data and be able to repopulate the database after cleaning the data.
+* Data Access Strategy
+ * Reasoning: Creating daily blockers
+* Admin Console Improvements
+ * Reasoning: Trigger has been met to refine tickets (Research Synthesis)
+* Improved Dev Tooling
+ * improve test\_parse.py 2641
+ * separate celery 2592
+ * Reasoning: Developing / refining tickets to support the above priorities – These tickets will enhance capabilities while the above are being flushed out
+
+### Sprint Goal
+
+**Dev:**
+
+_**Reparsing, Admin Console Improvements, Application Health Monitoring work, and Improved Dev Tooling**_
+
+* \#2965 — As tech lead, I want a database seed implemented for testing
+* \#3102 — Admin Exp: Django Implement Multi-Select Fiscal Period Dropdown For Data Export
+* \#2561 — As a sys admin, I need TDP to automatically deactivate accounts that are inactive for 180 days
+* \#3110 — Spike - Investigate Custom Filter Integration
+* \#3137 — \[bug] OFA unable to export data to csv by record type and fiscal period
+* \#3074 — TDP Data Files page permissions for DIGIT & Sys Admin user groups
+* \#3076 — Admin Filter Enhancements for Data Files Page
+
+**DevOps:**
+
+_**Successful deployments across environments and pipeline stability investments**_
+
+*
+
+**Design:**
+
+_**Support reviews, In-app banner to support parsed data, Continue Error Audit (Cat 4)**_
+
+* \#2968 — \[Design Deliverable] Update Error Audit for Cat 4 / QA
+* \#3114 — \[Design Spike] In-app banner for submission history pages w/ data parsed before May 2024
+* \#3143 — August release notes — Knowledge Center & Email Template
+
+
+
+## Tickets
+
+### Completed/Merged
+
+* [#2985 \[Design Deliverable\] Email spec for Admin Notification for stuck files](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2985)
+* [#2996 Add dynamic field name to cat4 error messages](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2996)
+* [#3143 August release notes — Knowledge Center & Email Template](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3143)
+* [#3061 \[a11y fix\] Django multi-select filter ](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3061)
+* [#2954 Extend SESSION\_COOKIE\_AGE](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2954)
+* [#3079 DB Backup Script Fix](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3079)
+
+### Submitted (QASP Review, OCIO Review)
+
+* [#3064 Re-parse Meta Model](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3064)
+* [#3065 Spike - Guarantee Sequential Execution of Re-parse Command](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3065)
+* [#2792 \[Error Audit\] Category 3 error messages clean-up](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2792)
+* [#2883 Pre-Made Reporting Dashboards on Kibana](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2883)
+* [#2561 As a sys admin, I need TDP to automatically deactivate accounts that are inactive for 180 days](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2561)
+* [#3078 \[Research Synthesis\] DIGIT Admin Experience Improvements](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3078)
+
+### Ready to Merge
+
+### Closed (Not Merged)
+
+* [#3147 S3 buckets contain fewer datafiles than DAC](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3147)
+
+### Moved to Next Sprint
+
+**In Progress**
+
+* [#2965 As tech lead, I want a database seed implemented for testing](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2965)
+* [#2458 Integrate Nexus into CircleCI](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2458)
+* [#3137 \[bug\] OFA unable to export data to csv by record type and fiscal period](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3137)
+* [#2968 \[Design Deliverable\] Update Error Audit for Cat 4 / QA](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/2968)
+* [#3060 As a TDP user, I need to stay logged in when I'm actively using the system](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3060)
+* [#3074 TDP Data Files page permissions for DIGIT & Sys Admin user groups](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3074)
+* [#3114 \[Design Spike\] In-app banner for submission history pages w/ data parsed before May 2024](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3114)
+* [#3142 \[Research Spike\] Get more detail about Yun & DIGIT's data workflow and use cases](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3142)
+
+#### Blocked
+
+*
+
+**Raft Review**
+
+* [#3043 Sentry: Local environment for Debugging](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3043)
+* [#3110 Spike - Investigate Custom Filter Integration](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3110)
+* [#3102 Admin Exp: Django Implement Multi-Select Fiscal Period Dropdown For Data Export ](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3102)
+* [#3087 Admin By Newest Filter Enhancements for Data Files Page](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3087)
+* [#3076 Admin Filter Enhancements for Data Files Page ](https://app.zenhub.com/workspaces/sprint-board-5f18ab06dfd91c000f7e682e/issues/gh/raft-tech/tanf-app/3076)
diff --git a/docs/Sprint-Review/sprint-107-summary.md b/docs/Sprint-Review/sprint-107-summary.md
new file mode 100644
index 000000000..e6d1aa4d9
--- /dev/null
+++ b/docs/Sprint-Review/sprint-107-summary.md
@@ -0,0 +1,89 @@
+# sprint-107-summary
+
+8/28/2024 - 9/10/2024
+
+### Priority Setting
+
+* Re-parsing epic
+* Postgres db access
+* UX research with DIGIT team
+* Continuous communication with STTs about latest TDP features and updates
+
+### Sprint Goal
+
+**Dev:**
+
+_**Re-parsing, Admin Console Improvements, and Application Health Monitoring work**_
+
+* \#3106 — Re-Parse Django Action
+* \#3137 — \[bug] OFA unable to export data to csv by record type and fiscal period
+* \#3074 — TDP Data Files page permissions for DIGIT & Sys Admin user groups
+* \#3044 — Prometheus/Grafana - Local Environment
+* \#3042 — Sentry in cloud.gov
+
+**DevOps:**
+_**Successful deployments across environments and pipeline stability investments**_
+
+* \#2965 — As tech lead, I want a database seed implemented for testing
+* \#2458 — Integrate Nexus into CircleCI
+
+**Design:**
+
+_**Support reviews, In-app banner to support parsed data, Continue Error Audit (Cat 4)**_
+
+* \#3156 — Release Notes Email Template
+* \#3100 — \[Design Deliverable] Update stakeholders & personas document
+* \#2968 — \[Design Deliverable] Update Error Audit for Cat 4 / QA
+
+## Tickets
+
+### Completed/Merged
+
+* [#2561 As a sys admin, I need TDP to automatically deactivate accounts that are inactive for 180 days](https://github.com/raft-tech/TANF-app/issues/2561)
+* [#2792 \[Error Audit\] Category 3 error messages clean-up ](https://github.com/raft-tech/TANF-app/issues/2792)
+* [#3043 Sentry: Local environment for Debugging](https://github.com/raft-tech/TANF-app/issues/3043)
+* [#3064 Re-parse Meta Model](https://github.com/raft-tech/TANF-app/issues/3064)
+* [#3065 Spike - Guarantee Sequential Execution of Re-parse Command](https://github.com/raft-tech/TANF-app/issues/3065)
+* [#3074 TDP Data Files page permissions for DIGIT & Sys Admin user groups ](https://github.com/raft-tech/TANF-app/issues/3074)
+* [#3076 Admin Filter Enhancements for Data Files Page ](https://github.com/raft-tech/TANF-app/issues/3076)
+* [#3078 \[Research Synthesis\] DIGIT Admin Experience Improvements ](https://github.com/raft-tech/TANF-app/issues/3078)
+* [#3087 Admin By Newest Filter Enhancements for Data Files Page ](https://github.com/raft-tech/TANF-app/issues/3087)
+* [#3114 \[Design Spike\] In-app banner for submission history pages w/ data parsed before May 2024 ](https://github.com/raft-tech/TANF-app/issues/3114)
+* [#3142 \[Research Spike\] Get more detail about Yun & DIGIT's data workflow and use cases ](https://github.com/raft-tech/TANF-app/issues/3142)
+
+### Submitted (QASP Review, OCIO Review)
+
+*
+
+### Ready to Merge
+
+* [#2883 Pre-Made Reporting Dashboards on Kibana ](https://github.com/raft-tech/TANF-app/issues/2883)
+* [#3102 Admin Exp: Django Implement Multi-Select Fiscal Period Dropdown For Data Export ](https://github.com/raft-tech/TANF-app/issues/3102)
+
+### Closed (Not Merged)
+
+* [#3110 Spike - Investigate Custom Filter Integration ](https://github.com/raft-tech/TANF-app/issues/3110)
+* [#3156 Release Notes Knowledge Center and Email Template ](https://github.com/raft-tech/TANF-app/issues/3156)
+
+### Moved to Next Sprint
+
+**In Progress**
+
+* [#2968 \[Design Deliverable\] Update Error Audit for Cat 4 / QA ](https://github.com/raft-tech/TANF-app/issues/2968)
+* [#3060 As a TDP user, I need to stay logged in when I'm actively using the system ](https://github.com/raft-tech/TANF-app/issues/3060)
+* [#3100 \[Design Deliverable\] Update stakeholders & personas document ](https://github.com/raft-tech/TANF-app/issues/3100)
+* [#3106 Re-Parse Django Action ](https://github.com/raft-tech/TANF-app/issues/3106)
+* [#3137 \[bug\] OFA unable to export data to csv by record type and fiscal period ](https://github.com/raft-tech/TANF-app/issues/3137)
+* [#3164 \[Research Synthesis\] Yun & DIGIT's data workflow and use cases ](https://github.com/raft-tech/TANF-app/issues/3164)
+* [#3170 Reparse Command Fails when Queryset is Large ](https://github.com/raft-tech/TANF-app/issues/3170)
+* [#3179 Spike - How We Work / Hopes & Fears Workshop prep ](https://github.com/raft-tech/TANF-app/issues/3179)
+
+**Blocked**
+
+*
+
+**Raft Review**
+
+* [#2458 Integrate Nexus into CircleCI ](https://github.com/raft-tech/TANF-app/issues/2458)
+* [#2965 As tech lead, I want a database seed implemented for testing ](https://github.com/raft-tech/TANF-app/issues/2965)
+* [#3044 Prometheus/Grafana - Local Environment ](https://github.com/raft-tech/TANF-app/issues/3044)
diff --git a/docs/Technical-Documentation/tech-memos/multi-select-fiters/multi-select-filters.md b/docs/Technical-Documentation/tech-memos/multi-select-fiters/multi-select-filters.md
new file mode 100644
index 000000000..93d668bd6
--- /dev/null
+++ b/docs/Technical-Documentation/tech-memos/multi-select-fiters/multi-select-filters.md
@@ -0,0 +1,171 @@
+# Multi-Select Filters
+
+**Audience**: TDP Software Engineers
+**Subject**: Multi-Select Filter Integration
+**Date**: August 27, 2024
+
+## Summary
+This technical memorandum provides the suggested guidelines for a future engineer to integrate TDP's need for multi-select filtering with Django 508. The memorandum provides some necessary background on both the TDP multi-select filters as well as Django 508 and it's purpose and effects. The [Method](#method) section provides the guidelines and updates required to integrate TDP's custom filtering needs with Django 508. Specifically, the [Django 508 Updates](#django-508-updates) section introduces the engineer to the area where the filtering and query string building occurs within Django 508 and the suggested changes. The [TDP Updates](#tdp-updates) section introduces the recommended changes to current TDP custom filtering with respect to how it can be simplified and how it could be unified with Django 508 to provide a seamless filtering experience.
+
+## Background
+TDP has been expanding it's Django Admin Console (DAC) filtering capabilities by introducing custom filters, specifically multi-select filters. This has introduced a myriad of issues because TDP does not use the default DAC. Instead, to assist with accessibility compliance TDP wraps the default DAC with [Django 508](https://github.com/raft-tech/django-admin-508) (henceforth referred to as 508) which makes various updates to the styling and functionality of the default DAC. A key change is that 508 introduces to the DAC is an `Apply Filters` button that intercepts query string parameters from default DAC filters and only applies them after clicking the button. The default DAC applies the filters as they are selected as opposed to all at once. The issue with 508's approach is that it assumes all filters are built-in Django filters (i.e. single select filters). This presents a discrepancy because Django allows developers to write custom templates and filters to add further filtering functionality (e.g. multi-select filters).
+
+## Out of Scope
+General filter template specification and general property based multi-select filtering mentioned in the [TDP Updates](#tdp-updates) section of this memorandum are out of scope for this memorandum.
+
+## Method
+To support multi-select/custom filtering in the DAC, both the TDP repository and the 508 repository will require updates.
+
+### Django 508 Updates
+508 builds the query string for all filters on the currently selected DAC page with the [dropdown-filter.js](https://github.com/raft-tech/django-admin-508/blob/main/admin_interface/static/admin_interface/508/dropdown-filter.js) JavaScript file. This file defines a JQuery function that operates on the `changelist-filter` element in the DOM. The function adds `onchange` event handlers to each filter in the `changelist-filter` element which extract the filter's query string template value that gets selected when it changes to construct a new query string. However, when custom templates and custom filters are introduced the JQuery function breaks down and cannot handle it. The implementation of the single-select and multi-select query building cannot be unified. This is because Django built-in single-select filters define a single prop on the `option` elements for the filter. The `value` prop that is defined on all the `option` elements is that `option`'s query parameter with the rest of the current query string appended to it. This same implementation cannot be achieved on multi-select filters because the query string cannot (and should not) contain multiple queries of the same type. This implies single-select and multi-select filters have to be handled in 508 differently. The update to `dropdown-filter.js` provided below serves as a guide towards a final solution for integrating multi-select filters, single-select filters, and the `Apply Filters` button. The implementation below relies on two key facts. One, all multi-select filters define `ariaMultiSelectable` and two, that all multi-select filters define two custom props: `key` and `value`. These key value pairs (e.g. `key=name__in`, `value=Bob`) are used to build the query string along with whatever the remaining single-select filters have chosen. When a user clicks the `Apply Filters` button, the code below executes and builds the query string for single and multi-select filters.
+
+```javascript
+if (typeof (django) !== 'undefined' && typeof (django.jQuery) !== 'undefined') {
+ (function ($) {
+ 'use strict';
+ $(document).ready(function () {
+ const filters = document.querySelectorAll('#changelist-filter .list-filter-dropdown select')
+ let query = '?'
+
+ const applyFiltersButton = document.querySelector('#submit-filters');
+ if (applyFiltersButton) {
+ applyFiltersButton.onclick = function () {
+ for (const filter of filters) {
+ let conjunction = query === '?' ? '' : '&'
+ if (!filter.ariaMultiSelectable) {
+ if (filter.selectedIndex !== 0) {
+ // Built in Django filters append the query string to the `value` field on the element. However, when we
+ // have a mult-selectable filter, the select element can't have the `value` field as a query string
+ // because multiple options can be selected and there is no way to track that. Therefore, we strip the
+ // single select filters query param from the existing query string and build and entirely new query
+ // string from that.
+ let opt = filter.options[filter.selectedIndex]
+ let query_str = opt.value
+ let filter_query = ''
+ for (let i = 1; i < query_str.length; i++) {
+ if (query_str[i] === '&') {
+ break
+ }
+ filter_query += query_str[i]
+ }
+ query = query.concat(conjunction, filter_query)
+ }
+ }
+ else {
+ // All multi select filters are required to set the `key` and `value` fields on the option element to the
+ // individual options to be able to build the correct query string.
+ let selected = ''
+ for (const option of filter.options) {
+ if (option.selected) {
+ selected = selected.concat(option.value, '%2C')
+ }
+ }
+ selected = selected.substring(0, selected.lastIndexOf('%2C'))
+ if (selected !== '') {
+ query = query.concat(conjunction, filter.options[0].getAttribute('key'), '=', selected)
+ }
+ }
+ }
+ window.location = query
+ };
+ }
+ });
+ })(django.jQuery);
+}
+```
+
+### TDP Updates
+Currently, TDP implements a custom multi-select filter, with a custom template. This filter is complex and relies on a custom "filter" button to apply it's selection which is very incohesive with the `Apply Filters` button that 508 introduces. To remedy this, the current and future multi-select/custom filters implemented in TDP need to give 508 control of constructing the appropriate query string by way of supplying 508 with the appropriate key-value pairs needed in their templates. In doing so, we can also simplify and generalize the current multi-select filter available in TDP.
+
+TDP currently utilizes three classes to implement field based multi-select filtering. This should be able to be simplified to a single class when we let 508 manage query string building. There are a few main features to note that this type of class would need. The first is the custom template used to create a multi-select drop down filter. An example template is the [multiselectdropdownfilter.html](multiselectdropdownfilter.html). The second are the unique query string parameters that would need to be defined in the `choices` method of the class. E.g the `key` and `value` parameters. These are the parameters that Django would populate into the aforementioned template and that 508 will need to parse to build the appropriate query string. Below is an example class that allows for field based multi-select filtering. The example class introduces some extra caveats to handle multi-select functionality. The `FieldListMultiSelectFilter` overrides and adds new parameters in the constructor before calling `super()`. These additions and overrides help convert the parent class `AllValuesFieldListFilter` from a single-select to a multi-select filter. Looking forward towards the future of TDP filtering, the example class implementation provides a path forward for Django model property based multi-select filtering (e.g. the `fiscal_period` property on the `DataFile` model). Leveraging the aforementioned template and building a class which sub-classes the Django `SimpleListFiter` class we have the ability to provide general multi-select filtering for Django model properties by implementing the correct `queryset`, `choices`, and `lookups` methods.
+
+```python
+class FieldListMultiSelectFilter(admin.AllValuesFieldListFilter):
+ """Multi select dropdown filter for all kind of fields."""
+
+ template = 'multiselectdropdownfilter.html'
+
+ def __init__(self, field, request, params, model, model_admin, field_path):
+ self.lookup_kwarg = '%s__in' % field_path
+ self.lookup_kwarg_isnull = '%s__isnull' % field_path
+ lookup_vals = request.GET.get(self.lookup_kwarg)
+ self.lookup_vals = lookup_vals.split(',') if lookup_vals else list()
+ self.lookup_val_isnull = request.GET.get(self.lookup_kwarg_isnull)
+ self.empty_value_display = model_admin.get_empty_value_display()
+ parent_model, reverse_path = reverse_field_path(model, field_path)
+ # Obey parent ModelAdmin queryset when deciding which options to show
+ if model == parent_model:
+ queryset = model_admin.get_queryset(request)
+ else:
+ queryset = parent_model._default_manager.all()
+ self.lookup_choices = (queryset
+ .distinct()
+ .order_by(field.name)
+ .values_list(field.name, flat=True))
+ super(admin.AllValuesFieldListFilter, self).__init__(field, request, params, model, model_admin, field_path)
+
+ def queryset(self, request, queryset):
+ """Build queryset based on choices."""
+ params = Q()
+ for lookup_arg, value in self.used_parameters.items():
+ params |= Q(**{lookup_arg: value})
+ try:
+ return queryset.filter(params)
+ except (ValueError, ValidationError) as e:
+ # Fields may raise a ValueError or ValidationError when converting
+ # the parameters to the correct type.
+ raise IncorrectLookupParameters(e)
+
+ def prepare_querystring_value(self, value):
+ """Mask commas."""
+ return str(value).replace(',', '%~')
+
+ def choices(self, changelist):
+ """Generate choices."""
+ add_facets = getattr(changelist, "add_facets", False)
+ facet_counts = self.get_facet_queryset(changelist) if add_facets else None
+ query_string = changelist.get_query_string({}, [self.lookup_kwarg, self.lookup_kwarg_isnull])
+ yield {
+ 'selected': not self.lookup_vals and self.lookup_val_isnull is None,
+ 'query_string': query_string,
+ 'display': _('All'),
+ }
+ include_none = False
+ count = None
+ empty_title = self.empty_value_display
+ for i, val in enumerate(self.lookup_choices):
+ if add_facets:
+ count = facet_counts[f"{i}__c"]
+ if val is None:
+ include_none = True
+ empty_title = f"{empty_title} ({count})" if add_facets else empty_title
+ continue
+
+ val = str(val)
+ qval = self.prepare_querystring_value(val)
+ yield {
+ 'selected': qval in self.lookup_vals,
+ 'query_string': query_string,
+ "display": f"{val} ({count})" if add_facets else val,
+ 'value': urllib.parse.quote_plus(val),
+ 'key': self.lookup_kwarg,
+ }
+ if include_none:
+ yield {
+ 'selected': bool(self.lookup_val_isnull),
+ 'query_string': query_string,
+ "display": empty_title,
+ 'value': 'True',
+ 'key': self.lookup_kwarg_isnull,
+ }
+```
+
+
+## Affected Systems
+- Django 508
+- TANF-App
+
+## Use and Test cases to consider
+- Consider adding 508 integration tests for all/most Django fields for the suggested `FieldListMultiSelectFilter`
+- Test having multiple Django built-in and `FieldListMultiSelectFilter`'s on the same page and verify the query string
+
diff --git a/docs/Technical-Documentation/tech-memos/multi-select-fiters/multiselectdropdownfilter.html b/docs/Technical-Documentation/tech-memos/multi-select-fiters/multiselectdropdownfilter.html
new file mode 100644
index 000000000..4cec46c77
--- /dev/null
+++ b/docs/Technical-Documentation/tech-memos/multi-select-fiters/multiselectdropdownfilter.html
@@ -0,0 +1,18 @@
+{% load i18n admin_urls %}
+