diff --git a/pyproject.toml b/pyproject.toml
index 999ec962..4eb18d64 100644
--- a/pyproject.toml
+++ b/pyproject.toml
@@ -97,32 +97,6 @@ dev-dependencies = [
"ruff>=0.9.3",
]
-[tool.black]
-line-length = 120
-include = '\.pyi?$'
-exclude = '''
-/(
- \.git
- | \.pytest_cache
- | \.tox
- | \.venv
- | ~build
- | build
- | ops
- | migrations
-)/
-'''
-
-#[tool.isort]
-#profile = "black"
-#line_length = 120
-#default_section = "THIRDPARTY"
-#known_first_party = []
-#known_django = "django"
-#sections = ["FUTURE","STDLIB","DJANGO","THIRDPARTY","FIRSTPARTY","LOCALFOLDER"]
-#include_trailing_comma = true
-#skip = ["migrations", "snapshots", ".venv"]
-
[tool.django-stubs]
django_settings_module = "country_workspace.config.settings"
diff --git a/ruff.toml b/ruff.toml
index ac22b1d1..6e47c266 100644
--- a/ruff.toml
+++ b/ruff.toml
@@ -80,8 +80,7 @@ force-wrap-aliases = true
combine-as-imports = true
extra-standard-library = ["path"]
known-third-party = ['django']
-known-first-party = ['hope_*']
-known-local-folder = ["country_workspace"]
+known-first-party = ['country_workspace']
no-sections = false
lines-after-imports = 1
relative-imports-order = "furthest-to-closest"
diff --git a/src/country_workspace/apps.py b/src/country_workspace/apps.py
index a8e89256..0bb5d8f4 100644
--- a/src/country_workspace/apps.py
+++ b/src/country_workspace/apps.py
@@ -15,6 +15,7 @@ def ready(self) -> None:
import django_celery_boost.admin
import country_workspace.compat.admin_extra_buttons as c
+
from .utils import flags # noqa
admin_extra_buttons.api.confirm_action = c.confirm_action
diff --git a/src/country_workspace/cache/manager.py b/src/country_workspace/cache/manager.py
index 45601bbf..6916fc2d 100644
--- a/src/country_workspace/cache/manager.py
+++ b/src/country_workspace/cache/manager.py
@@ -10,10 +10,11 @@
from redis_lock.django_cache import RedisCache
from sentry_sdk import capture_exception
-from .signals import cache_get, cache_invalidate, cache_set
from country_workspace import VERSION
from country_workspace.state import state
+from .signals import cache_get, cache_invalidate, cache_set
+
if TYPE_CHECKING:
from ..models import Office, Program
logger = logging.getLogger(__name__)
diff --git a/src/country_workspace/contrib/hope/client.py b/src/country_workspace/contrib/hope/client.py
index e48b63c8..101d9eac 100644
--- a/src/country_workspace/contrib/hope/client.py
+++ b/src/country_workspace/contrib/hope/client.py
@@ -8,9 +8,10 @@
from constance import config
from requests.exceptions import RequestException
-from .signals import hope_request_end, hope_request_start
from country_workspace.exceptions import RemoteError
+from .signals import hope_request_end, hope_request_start
+
if TYPE_CHECKING:
JsonType = None | int | str | bool | list["JsonType"] | dict[str, "JsonType"]
FlatJsonType = dict[str, str | int | bool]
diff --git a/src/country_workspace/contrib/hope/geo.py b/src/country_workspace/contrib/hope/geo.py
index 905b75dd..e8f8813d 100644
--- a/src/country_workspace/contrib/hope/geo.py
+++ b/src/country_workspace/contrib/hope/geo.py
@@ -4,12 +4,12 @@
from django import forms
from django.core.exceptions import ValidationError
from django.utils.text import slugify
-
from hope_flex_fields.mixin import ChildFieldMixin
+from country_workspace.cache.manager import cache_manager
+
from ...exceptions import RemoteError
from .client import HopeClient
-from country_workspace.cache.manager import cache_manager
logger = logging.getLogger(__name__)
diff --git a/src/country_workspace/contrib/hope/remotes/country.py b/src/country_workspace/contrib/hope/remotes/country.py
index 73c3eb16..6ade2983 100644
--- a/src/country_workspace/contrib/hope/remotes/country.py
+++ b/src/country_workspace/contrib/hope/remotes/country.py
@@ -1,7 +1,6 @@
from typing import TYPE_CHECKING
from django import forms
-
from hope_flex_fields.attributes.abstract import AbstractAttributeHandler, AttributeHandlerConfig
if TYPE_CHECKING:
diff --git a/src/country_workspace/contrib/hope/sync/office.py b/src/country_workspace/contrib/hope/sync/office.py
index 54760b8b..ddd859f2 100644
--- a/src/country_workspace/contrib/hope/sync/office.py
+++ b/src/country_workspace/contrib/hope/sync/office.py
@@ -1,12 +1,12 @@
from io import TextIOBase
from django.core.cache import cache
-
from hope_flex_fields.models import DataChecker
+from country_workspace.models import Office, Program, SyncLog
+
from .. import constants
from ..client import HopeClient
-from country_workspace.models import Office, Program, SyncLog
def sync_offices(stdout: TextIOBase | None = None) -> dict[str, int]:
diff --git a/src/country_workspace/datasources/rdi.py b/src/country_workspace/datasources/rdi.py
index fb350777..dce5a9ed 100644
--- a/src/country_workspace/datasources/rdi.py
+++ b/src/country_workspace/datasources/rdi.py
@@ -1,7 +1,6 @@
import io
from django.db.transaction import atomic
-
from hope_smart_import.readers import open_xls_multi
from country_workspace.models import AsyncJob, Batch, Household
diff --git a/src/country_workspace/models/base.py b/src/country_workspace/models/base.py
index edd2805c..5fcfdbca 100644
--- a/src/country_workspace/models/base.py
+++ b/src/country_workspace/models/base.py
@@ -14,7 +14,6 @@
if TYPE_CHECKING:
from django.db.models import QuerySet
-
from hope_flex_fields.models import DataChecker
from country_workspace.models import Office, Program
diff --git a/src/country_workspace/models/household.py b/src/country_workspace/models/household.py
index c1233786..a29b5b2d 100644
--- a/src/country_workspace/models/household.py
+++ b/src/country_workspace/models/household.py
@@ -3,12 +3,12 @@
import reversion
from django.db import models
+from django.utils import timezone
from .base import BaseModel, Validable
if TYPE_CHECKING:
from django.db.models import QuerySet
-
from hope_flex_fields.models import DataChecker
from .individual import Individual
@@ -37,11 +37,19 @@ def country_office(self) -> "Office":
return self.batch.program.country_office
def validate_with_checker(self) -> bool:
- super().validate_with_checker()
- errors = self.program.beneficiary_validator.validate(self)
- if errors:
- self.errors["dct"] = errors
- self.save(update_fields=["errors"])
+ hh_valid = True
+ for ind in self.members.all():
+ if not ind.validate_with_checker():
+ hh_valid = False
+ if hh_valid:
+ super().validate_with_checker()
+ errors = self.program.beneficiary_validator.validate(self)
+ if errors:
+ self.errors["dct"] = errors
+ else:
+ self.errors["dct"] = ["Some member did not validate"]
+ self.last_checked = timezone.now()
+ self.save(update_fields=["errors", "last_checked"])
return not bool(self.errors)
# Business methods
diff --git a/src/country_workspace/models/individual.py b/src/country_workspace/models/individual.py
index 3dbd98f5..a39785d5 100644
--- a/src/country_workspace/models/individual.py
+++ b/src/country_workspace/models/individual.py
@@ -22,8 +22,8 @@ def checker(self) -> "DataChecker":
@cached_property
def program(self) -> "DataChecker":
- return self.household.batch.program
+ return self.batch.program
@cached_property
def country_office(self) -> "DataChecker":
- return self.household.batch.program.country_office
+ return self.batch.program.country_office
diff --git a/src/country_workspace/models/program.py b/src/country_workspace/models/program.py
index ff589b12..e39ab30e 100644
--- a/src/country_workspace/models/program.py
+++ b/src/country_workspace/models/program.py
@@ -2,14 +2,14 @@
from django.db import models
from django.utils.translation import gettext as _
+from hope_flex_fields.models import DataChecker
from strategy_field.fields import StrategyField
from strategy_field.utils import fqn
-from hope_flex_fields.models import DataChecker
+from country_workspace.models.office import Office
from ..validators.registry import NoopValidator, beneficiary_validator_registry
from .base import BaseModel, Validable
-from country_workspace.models.office import Office
if TYPE_CHECKING:
from django.db.models import QuerySet
diff --git a/src/country_workspace/tasks.py b/src/country_workspace/tasks.py
index a3d4fa14..7288071c 100644
--- a/src/country_workspace/tasks.py
+++ b/src/country_workspace/tasks.py
@@ -39,7 +39,7 @@ def sync_job_task(pk: int, version: int) -> dict[str, Any]:
pk=pk,
version=version,
)
- except AsyncJob.DoesNotExist as e:
+ except AsyncJob.DoesNotExist as e: # pragma: no cover
sentry_sdk.capture_exception(e)
raise e
diff --git a/src/country_workspace/versioning/checkers.py b/src/country_workspace/versioning/checkers.py
index b2346436..760f2b84 100644
--- a/src/country_workspace/versioning/checkers.py
+++ b/src/country_workspace/versioning/checkers.py
@@ -1,5 +1,4 @@
from django import forms
-
from hope_flex_fields.models import DataChecker, FieldDefinition, Fieldset
from country_workspace.contrib.hope.constants import HOUSEHOLD_CHECKER_NAME, INDIVIDUAL_CHECKER_NAME
diff --git a/src/country_workspace/versioning/hope_fields.py b/src/country_workspace/versioning/hope_fields.py
index 1be3b381..b8b0b07e 100644
--- a/src/country_workspace/versioning/hope_fields.py
+++ b/src/country_workspace/versioning/hope_fields.py
@@ -1,7 +1,6 @@
from django import forms
from django.conf import settings
from django.utils.text import slugify
-
from hope_flex_fields.models import FieldDefinition
from country_workspace.models import SyncLog
diff --git a/src/country_workspace/versioning/scripts/0001_initial.py b/src/country_workspace/versioning/scripts/0001_initial.py
index ac2c5a89..2c0e1901 100644
--- a/src/country_workspace/versioning/scripts/0001_initial.py
+++ b/src/country_workspace/versioning/scripts/0001_initial.py
@@ -1,6 +1,5 @@
# Generated by Django 5.1.1 on 2024-10-09 06:19
from flags.state import enable_flag
-
from hope_flex_fields.models import DataChecker, Fieldset
from country_workspace.contrib.hope.constants import HOUSEHOLD_CHECKER_NAME, INDIVIDUAL_CHECKER_NAME
diff --git a/src/country_workspace/workspaces/admin/cleaners/actions.py b/src/country_workspace/workspaces/admin/cleaners/actions.py
index 3d0315da..f181734d 100644
--- a/src/country_workspace/workspaces/admin/cleaners/actions.py
+++ b/src/country_workspace/workspaces/admin/cleaners/actions.py
@@ -6,14 +6,15 @@
from django.utils.translation import gettext as _
from strategy_field.utils import fqn
+from country_workspace.models import AsyncJob
+from country_workspace.state import state
+from country_workspace.workspaces.admin.forms import BulkUpdateExportForm
+
from .bulk_update import bulk_update_export_template
from .calculate_checksum import calculate_checksum_impl
from .mass_update import MassUpdateForm, mass_update_impl
from .regex import RegexUpdateForm, regex_update_impl
from .validate import validate_queryset
-from country_workspace.models import AsyncJob
-from country_workspace.state import state
-from country_workspace.workspaces.admin.forms import BulkUpdateExportForm
if TYPE_CHECKING:
from django.db.models import QuerySet
diff --git a/src/country_workspace/workspaces/admin/cleaners/bulk_update.py b/src/country_workspace/workspaces/admin/cleaners/bulk_update.py
index 4a7f7c00..a6d151d1 100644
--- a/src/country_workspace/workspaces/admin/cleaners/bulk_update.py
+++ b/src/country_workspace/workspaces/admin/cleaners/bulk_update.py
@@ -6,11 +6,10 @@
from django import forms
from django.apps import apps
from django.core.exceptions import ObjectDoesNotExist
-from xlsxwriter import Workbook
-
from hope_flex_fields.models import DataChecker, FlexField
from hope_flex_fields.xlsx import get_format_for_field
from hope_smart_import.readers import open_xls
+from xlsxwriter import Workbook
from country_workspace.models import AsyncJob, Program
from country_workspace.storages import MEDIA_STORAGE
diff --git a/src/country_workspace/workspaces/admin/cleaners/mass_update.py b/src/country_workspace/workspaces/admin/cleaners/mass_update.py
index 82b156d1..c3354e3a 100644
--- a/src/country_workspace/workspaces/admin/cleaners/mass_update.py
+++ b/src/country_workspace/workspaces/admin/cleaners/mass_update.py
@@ -4,15 +4,13 @@
from django.db import transaction
from django.forms import MultiValueField, widgets
from django.utils.text import slugify
-from strategy_field.utils import fqn
-
from hope_flex_fields.fields import FlexFormMixin
+from strategy_field.utils import fqn
from .base import BaseActionForm
if TYPE_CHECKING:
from django.db.models import QuerySet
-
from hope_flex_fields.models import DataChecker
from country_workspace.types import Beneficiary
diff --git a/src/country_workspace/workspaces/admin/cleaners/regex.py b/src/country_workspace/workspaces/admin/cleaners/regex.py
index 8e43e743..af5a8659 100644
--- a/src/country_workspace/workspaces/admin/cleaners/regex.py
+++ b/src/country_workspace/workspaces/admin/cleaners/regex.py
@@ -4,12 +4,12 @@
from django import forms
from django.db import transaction
-from .base import BaseActionForm
from country_workspace.utils.flex_fields import get_checker_fields
+from .base import BaseActionForm
+
if TYPE_CHECKING:
from django.db.models import QuerySet
-
from hope_flex_fields.models import DataChecker
from country_workspace.types import Beneficiary
diff --git a/src/country_workspace/workspaces/admin/cleaners/validate.py b/src/country_workspace/workspaces/admin/cleaners/validate.py
index d4bed611..bad4f5f2 100644
--- a/src/country_workspace/workspaces/admin/cleaners/validate.py
+++ b/src/country_workspace/workspaces/admin/cleaners/validate.py
@@ -1,20 +1,44 @@
import logging
-from typing import Any
+from typing import TYPE_CHECKING, Any
from django.db.models import Model, QuerySet
+from country_workspace.models import AsyncJob, Household, Program
+
+if TYPE_CHECKING:
+ from country_workspace.models.base import Validable
+
logger = logging.getLogger(__name__)
def validate_queryset(queryset: QuerySet[Model], **kwargs: Any) -> dict[str, int]:
valid = invalid = num = 0
+ entry: "Validable"
try:
for __, entry in enumerate(queryset, 1):
if entry.validate_with_checker():
valid += 1
else:
invalid += 1
- except Exception as e:
+ except Exception as e: # pragma: no cover
+ logger.exception(e)
+ raise
+
+ return {"valid": valid, "invalid": invalid, "total": num}
+
+
+def validate_program(job: AsyncJob) -> dict[str, int]:
+ valid = invalid = num = 0
+ hh: Household
+ try:
+ p: Program = job.program
+ for hh in Household.objects.filter(batch__program=p):
+ if hh.validate_with_checker():
+ valid += 1
+ else:
+ invalid += 1
+ except Exception as e: # pragma: no cover
logger.exception(e)
+ raise
return {"valid": valid, "invalid": invalid, "total": num}
diff --git a/src/country_workspace/workspaces/admin/hh_ind.py b/src/country_workspace/workspaces/admin/hh_ind.py
index 1d53f5bd..8b77a596 100644
--- a/src/country_workspace/workspaces/admin/hh_ind.py
+++ b/src/country_workspace/workspaces/admin/hh_ind.py
@@ -18,7 +18,7 @@
from ...state import state
from ..options import WorkspaceModelAdmin
from .cleaners import actions
-from .cleaners.validate import validate_queryset
+from .cleaners.validate import validate_program
if TYPE_CHECKING:
from hope_flex_fields.forms import FlexForm
@@ -102,15 +102,14 @@ def validate_program(self, request: HttpRequest) -> "HttpResponse":
opts = self.model._meta
job = AsyncJob.objects.create(
description="Validate Program %s" % opts.proxy_for_model._meta.verbose_name_plural,
- type=AsyncJob.JobType.ACTION,
+ type=AsyncJob.JobType.TASK,
owner=state.request.user,
- action=fqn(validate_queryset),
+ action=fqn(validate_program),
program=state.program,
- config={"pks": "__all__", "model_name": opts.label},
+ config={},
)
job.queue()
self.message_user(request, "Task scheduled", messages.SUCCESS)
- return job
@button()
def view_raw_data(self, request: HttpRequest, pk: str) -> "HttpResponse":
diff --git a/src/country_workspace/workspaces/admin/program.py b/src/country_workspace/workspaces/admin/program.py
index b45b270b..37cf9eb3 100644
--- a/src/country_workspace/workspaces/admin/program.py
+++ b/src/country_workspace/workspaces/admin/program.py
@@ -13,6 +13,10 @@
from django.utils.translation import gettext as _
from strategy_field.utils import fqn
+from country_workspace.constants import BATCH_NAME_DEFAULT
+from country_workspace.contrib.aurora.pipeline import import_from_aurora
+from country_workspace.state import state
+
from ...contrib.aurora.forms import ImportAuroraForm
from ...contrib.kobo.forms import ImportKoboForm
from ...contrib.kobo.sync import import_data as import_from_kobo
@@ -24,9 +28,6 @@
from ..sites import workspace
from .cleaners.bulk_update import bulk_update_household, bulk_update_individual
from .forms import BulkUpdateImportForm, ImportFileForm
-from country_workspace.constants import BATCH_NAME_DEFAULT
-from country_workspace.contrib.aurora.pipeline import import_from_aurora
-from country_workspace.state import state
if TYPE_CHECKING:
from hope_flex_fields.models import DataChecker
diff --git a/src/country_workspace/workspaces/models.py b/src/country_workspace/workspaces/models.py
index c5839e4a..916e44c1 100644
--- a/src/country_workspace/workspaces/models.py
+++ b/src/country_workspace/workspaces/models.py
@@ -3,7 +3,6 @@
import reversion
from django.db import models
from django.utils.functional import cached_property
-
from hope_flex_fields.models import DataChecker
from country_workspace.models import AsyncJob, Batch, Household, Individual, Office, Program
diff --git a/src/country_workspace/workspaces/templates/workspace/hh_ind_change_form.html b/src/country_workspace/workspaces/templates/workspace/hh_ind_change_form.html
index 56802259..9db0733b 100644
--- a/src/country_workspace/workspaces/templates/workspace/hh_ind_change_form.html
+++ b/src/country_workspace/workspaces/templates/workspace/hh_ind_change_form.html
@@ -24,8 +24,12 @@
{{ adminform.form.non_field_errors }}
{% endif %}
{% if original.errors %}
-
Record does not validate and cannot be sent to HOPE
- {{ original.errors.dct }}
+ Record does not validate and cannot be sent to HOPE
+
+ {% for e in original.errors.dct %}
+ - {{ e }}
+ {% endfor %}
+
{% endif %}
{% block field_sets %}
diff --git a/src/country_workspace/workspaces/theme/static/css/styles.css b/src/country_workspace/workspaces/theme/static/css/styles.css
index 8358fe0d..7aecf728 100644
--- a/src/country_workspace/workspaces/theme/static/css/styles.css
+++ b/src/country_workspace/workspaces/theme/static/css/styles.css
@@ -1,2 +1,2 @@
-:root{--primary:#0f2f3b;--header-bg:#1e3a8a;--breadcrumbs-bg:#0f2f3b;--link-fg:#37b0f8;--accent:#f5dd5d;--header-color:#ffc;--border-color:#ccc;--body-bg:#fff;--darkened-bg:#f8f8f8;--selected-bg:#e4e4e4;--body-quiet-color:#666;--hairline-color:#e8e8e8}*{font-family:Roboto,Helvetica,Arial,sans-serif;color:rgb(0 0 0/var(--tw-text-opacity))}*,.breadcrumbs{--tw-text-opacity:1}.breadcrumbs{padding-top:.5rem;padding-bottom:.5rem;color:rgb(0 60 143/var(--tw-text-opacity));//line-height:24px}ul.object-tools{margin-top:1.25rem;margin-bottom:1.25rem;margin-left:.75rem;display:flex;padding-top:.25rem;padding-bottom:.25rem;padding-right:1.25rem;justify-content:flex-end}ul.object-tools li{margin-left:.25rem;margin-right:.25rem;font-size:.875rem;line-height:1.25rem;font-weight:400}ul.object-tools li a{margin-top:.5rem;margin-bottom:.5rem;--tw-bg-opacity:1;background-color:rgb(0 60 143/var(--tw-bg-opacity));padding:.25rem .75rem;--tw-text-opacity:1;color:rgb(255 255 255/var(--tw-text-opacity));margin:0;color:#fff;cursor:pointer;font-size:.875rem;vertical-align:middle;justify-content:center;min-width:64px;box-shadow:0 3px 1px -2px rgba(0,0,0,.2),0 2px 2px 0 rgba(0,0,0,.14),0 1px 5px 0 rgba(0,0,0,.12);outline:0;border:0 initial;border-color:initial;-o-border-image:initial;border-image:initial;text-decoration:none;padding:6px 16px;border-radius:4px;transition:background-color .25s cubic-bezier(.4,0,.2,1),box-shadow .25s cubic-bezier(.4,0,.2,1),border-color .25s cubic-bezier(.4,0,.2,1),color .25s cubic-bezier(.4,0,.2,1);background-color:#003c8f}ul.object-tools li a.disabled{border-width:1px;--tw-border-opacity:1;border-color:rgb(0 60 143/var(--tw-border-opacity))}.admin-change-link{font-size:.75rem;line-height:1rem}.admin-change-link,.admin-change-link .icon{--tw-text-opacity:1;color:rgb(0 173 239/var(--tw-text-opacity))}.admin-change-link .icon{vertical-align:middle}.messages{position:absolute;top:2.5rem;z-index:50;display:flex;align-items:center;border-radius:.25rem;--tw-border-opacity:1;border-color:rgb(0 0 0/var(--tw-border-opacity));padding-left:.5rem;padding-right:.5rem;--tw-drop-shadow:drop-shadow(0 4px 3px rgba(0,0,0,.07)) drop-shadow(0 2px 2px rgba(0,0,0,.06));filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.messages .message{top:2.5rem;width:100%;border-radius:.25rem;--tw-bg-opacity:1;background-color:rgb(248 113 113/var(--tw-bg-opacity));padding:1rem;font-weight:700}.messages .message.success{--tw-bg-opacity:1;background-color:rgb(134 239 172/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(22 101 52/var(--tw-text-opacity))}.messages .message.error{--tw-bg-opacity:1;background-color:rgb(248 113 113/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(127 29 29/var(--tw-text-opacity))}.messages .message.warn,.messages .message.warning{--tw-bg-opacity:1;background-color:rgb(253 224 71/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(113 63 18/var(--tw-text-opacity))}#content #content-main,#content #content-related{display:flex;--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity));padding:.75rem}#content #content-related{margin-left:.25rem;margin-right:.25rem}table th{text-align:left;font-weight:400}table td{padding:.25rem}table tr:nth-child(2n).error,table tr:nth-child(odd).error{--tw-bg-opacity:1;background-color:rgb(254 202 202/var(--tw-bg-opacity))}table tr:nth-child(2n){--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}#login-form{align-content:center;border-radius:.25rem;--tw-bg-opacity:1;background-color:rgb(209 213 219/var(--tw-bg-opacity));padding:1.25rem}#login-form table{min-width:100%}#login-form table tr:nth-child(2n),#login-form table tr:nth-child(odd){--tw-bg-opacity:1;background-color:rgb(209 213 219/var(--tw-bg-opacity));text-align:center}#login-form table tr:nth-child(2n) td input,#login-form table tr:nth-child(odd) td input{min-width:100%}#changelist{display:flex;align-items:flex-start;justify-content:space-between}#changelist .changelist-form-container{width:50%}#changelist .changelist-form-container #changelist-form .results{width:300px;overflow:scroll}#changelist .changelist-form-container #changelist-form .actions{margin-bottom:.5rem}#changelist .changelist-form-container #changelist-form .actions select{padding-top:.25rem;padding-bottom:.25rem;font-size:.875rem;line-height:1.25rem}#changelist .changelist-form-container #changelist-form .actions button[type=submit]{font-size:.875rem;line-height:1.25rem}#changelist .changelist-form-container #changelist-form .actions .action-counter,#changelist .changelist-form-container #changelist-form .actions .all{margin-left:1.25rem}#changelist .changelist-form-container #changelist-form .actions .clear a,#changelist .changelist-form-container #changelist-form .actions .question a{--tw-text-opacity:1;color:rgb(0 60 143/var(--tw-text-opacity))}#changelist .clear{clear:both}#changelist table{border-right:none;border-collapse:collapse;border-color:var(--border-color)}#changelist table thead{display:table-header-group;vertical-align:middle;unicode-bidi:isolate}#changelist table thead th{//background:var(--selected-bg);font-size:.6875rem;border:none;border-bottom:1px solid var(--hairline-color);border-top:1px solid var(--hairline-color);color:var(--body-quiet-color);display:table-cell;font-weight:500;line-height:normal;padding:0;text-align:left;text-transform:uppercase;vertical-align:middle;white-space:nowrap}#changelist table thead th.sorted .text{margin-right:42px}#changelist table thead th .text span{padding:8px 10px}#changelist table thead th .text a{display:block;cursor:pointer;padding:8px 10px;color:var(--body-quiet-color);text-decoration:none;transition:color .15s,background .15s}#changelist table thead th .sortoptions{padding:9px 5px 0;float:right;display:block;text-align:right}#changelist table thead th .sortoptions a{position:relative;width:14px;height:14px;display:inline-block;background:url(sorting-icons.svg) 0 0 no-repeat;background-size:14px auto}#changelist table thead th .sortoptions a.ascending{background-position:0 -28px}#changelist table thead th .sortoptions a.descending{top:1px;background-position:0 -56px}#changelist table thead th .sortoptions a:link{transition:color .15s,background .15s}#changelist table thead th .sortoptions a.sortremove{background-position:0 0}#changelist table thead th.action-checkbox-column{width:1.5em;text-align:center}#changelist table tbody tr:hover{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}#changelist table tbody tr{border-width:1px;--tw-border-opacity:1;border-bottom-color:rgb(243 244 246/var(--tw-border-opacity))}#changelist table tbody td,#changelist table tbody th{padding:20px;white-space:nowrap;font-size:.8125rem;overflow:auto;cursor:pointer}#changelist table tbody td.action-checkbox,#changelist table tbody th.action-checkbox{width:1.5em;text-align:center}#changelist table tbody tr:nth-child(odd){--tw-border-opacity:1;border-bottom-color:rgb(75 85 99/var(--tw-border-opacity));--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity));padding-top:3rem;padding-bottom:3rem}#changelist table tbody tr:nth-child(odd):hover{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}#changelist table tbody tr:nth-child(2n){--tw-border-opacity:1;border-bottom-color:rgb(75 85 99/var(--tw-border-opacity));--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity));padding-top:3rem;padding-bottom:3rem}#changelist table tbody tr:nth-child(2n):hover{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}#changelist table tbody tr.selected:nth-child(2n),#changelist table tbody tr.selected:nth-child(odd){--tw-bg-opacity:1;background-color:rgb(219 234 254/var(--tw-bg-opacity))}#changelist-filter{margin-left:.5rem;border-radius:.25rem;border-width:1px;--tw-border-opacity:1;border-color:rgb(107 114 128/var(--tw-border-opacity))}#changelist-filter h2{font-size:.875rem;text-transform:uppercase;letter-spacing:.5px;padding:5px 15px;margin-bottom:12px;border-bottom:none}#changelist-filter #changelist-filter-extra-actions a{font-family:ui-sans-serif,system-ui,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji;font-size:.875rem;line-height:1.25rem;font-style:italic}#changelist-filter h3{font-weight:400;padding:0 15px;margin-bottom:10px}#changelist-filter details{margin-top:.25rem}#changelist-filter details>*{display:inline}#changelist-filter details summary{font-weight:400;padding:0 15px;margin-bottom:10px;list-style-type:none;cursor:pointer}#changelist-filter details summary:before{//content:"→";content:"►";//content:"▷";font-weight:bolder;font-size:18px;font-family:Courier New,Courier,monospace;color:var(--link-hover-color)}#changelist-filter details:last-child{margin-bottom:1rem}#changelist-filter details.active summary{font-weight:700;--tw-text-opacity:1;color:rgb(0 173 239/var(--tw-text-opacity))}#changelist-filter details[open]>summary:before{//content:"↓";content:"▼"}#changelist-filter ul{margin:0;padding-left:.5rem;padding-right:.5rem}#changelist-filter ul select{min-width:100%;padding:.25rem;font-size:.875rem;line-height:1.25rem;border-radius:4px;min-height:30px;height:28px}#changelist-filter ul select,#changelist-filter ul select:active{border-color:#ccc}#changelist-filter ul li{min-width:100%}#changelist-filter ul li .select2-container{border-radius:4px;min-width:100%}#changelist-filter ul li.selected{font-weight:700;--tw-text-opacity:1;color:rgb(0 173 239/var(--tw-text-opacity))}#changelist-filter ul:last-child{border-bottom:none}table thead th.action-checkbox-column{width:1.5em;text-align:center}.paginator{display:flex;border-width:1px;--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity));padding:.5rem}.paginator a,.paginator span{margin-left:.25rem;margin-right:.25rem;border-width:1px;--tw-border-opacity:1;border-color:rgb(75 85 99/var(--tw-border-opacity));padding-left:.5rem;padding-right:.5rem}.paginator span{--tw-bg-opacity:1;background-color:rgb(156 163 175/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(255 255 255/var(--tw-text-opacity))}form input [type=date],form input[type=number],form input[type=password],form input[type=text],form select{border-radius:.25rem;border-width:1px;--tw-border-opacity:1;border-color:rgb(107 114 128/var(--tw-border-opacity));padding-top:.5rem;padding-bottom:.5rem;font-size:.75rem;line-height:1rem;font-weight:400}form .helptext{font-size:.75rem;line-height:1rem;font-style:italic}a.button,a.closelink,a.deletelink,button.button,input[type=button],input[type=submit]{margin:0;color:#fff;cursor:pointer;font-size:.875rem;vertical-align:middle;justify-content:center;min-width:64px;box-shadow:0 3px 1px -2px rgba(0,0,0,.2),0 2px 2px 0 rgba(0,0,0,.14),0 1px 5px 0 rgba(0,0,0,.12);outline:0;border:0 initial;border-color:initial;-o-border-image:initial;border-image:initial;text-decoration:none;line-height:inherit;height:33px;display:inline-block;padding:6px 16px;border-radius:4px;transition:background-color .25s cubic-bezier(.4,0,.2,1),box-shadow .25s cubic-bezier(.4,0,.2,1),border-color .25s cubic-bezier(.4,0,.2,1),color .25s cubic-bezier(.4,0,.2,1);background-color:#003c8f}a.button.default,a.closelink.default,a.deletelink.default,button.button.default,input[type=button].default,input[type=submit].default{background-color:#003c8f}a.button.closelink,a.closelink.closelink,a.deletelink.closelink,button.button.closelink,input[type=button].closelink,input[type=submit].closelink{background-color:rgba(99,92,15,.53)}.submit-row{margin-top:1.25rem}.submit-row a.closelink{@extend a.button;background-color:rgba(99,92,15,.53)}.submit-row a.deletelink{background-color:#cf2222;float:right}fieldset.module{border:none;margin:0 0 30px;background:var(--body-bg);min-width:0;padding:0}fieldset.module .fieldBox{margin-right:20px}fieldset.module .form-row{overflow:hidden;padding:10px;font-size:.8125rem;border-bottom:1px solid var(--hairline-color);margin-bottom:0}fieldset.module .form-row input,fieldset.module .form-row select{padding-top:.25rem;padding-bottom:.25rem;font-size:.875rem;line-height:1.25rem}fieldset.module .form-row .flex-container{display:flex}fieldset.module .form-row .flex-container .related-widget-wrapper{display:flex;gap:0 10px;flex-grow:1;flex-wrap:wrap;margin-bottom:5px}fieldset.module .form-row label{display:block;padding:4px 10px 0 0;min-width:160px;width:160px;word-wrap:break-word;color:var(--body-quiet-color);font-size:.8125rem}.beneficiary-form input[type=text],.beneficiary-form select{min-width:100%}.beneficiary-form .validation-error{font-style:italic}body.web #select2-id_program-results>li.select2-results__option.select2-results__option--highlighted,body.web li.select2-results__option.select2-results__option--highlighted{--tw-bg-opacity:1;background-color:rgb(0 173 239/var(--tw-bg-opacity))}body.web header{--tw-bg-opacity:1;background-color:rgb(35 57 68/var(--tw-bg-opacity));color:#fff}body.web header .select2{background-color:var(--body-bg)}body.web header .select2-container.select2-container--default{font-size:.875rem;line-height:1.25rem;min-width:600px;height:38px;border-radius:4px;border-width:0}body.web header .select2-container.select2-container--default .select2-results__option--highlighted[aria-selected]{--tw-bg-opacity:1;background-color:rgb(0 173 239/var(--tw-bg-opacity));color:#fff}body.web header .select2-container.select2-container--default .selection{font-size:.875rem;line-height:1.25rem;height:34px;border-radius:4px;line-height:34px;border-width:0}body.web header .select2-container.select2-container--default .selection .select2-selection.select2-selection--single{height:34px;border-radius:4px;border-width:0}body.web header .select2-container.select2-container--default .selection .select2-selection.select2-selection--single .select2-selection__rendered{line-height:38px;border-color:rgb(107 114 128/var(--tw-border-opacity));border-width:1px;border-radius:4px}body.web header .select2-container.select2-container--default .selection .select2-selection.select2-selection--single .select2-selection__arrow{height:38px;line-height:38px}body.web select{min-width:12rem;padding-top:.5rem;padding-bottom:.5rem;font-size:.875rem;line-height:1.25rem;border-radius:4px;border-width:1px;border-color:rgb(107 114 128/var(--tw-border-opacity));padding-right:2.5rem}body.web svg.sidebar-button{fill:#fff;width:1em;height:1em;display:inline-block;font-size:1.5rem;flex-shrink:0;-webkit-user-select:none;-moz-user-select:none;user-select:none;color:#fff;border-color:#fff}.title-top{box-shadow:0 2px 4px -1px rgba(0,0,0,.2),0 4px 5px 0 rgba(0,0,0,.14),0 1px 10px 0 rgba(0,0,0,.12);position:relative;width:100%;background-color:#fff}.title-top h2{font-size:1.5rem;font-weight:400;line-height:1.334;letter-spacing:0;color:rgba(0,0,0,.87);padding:28px 44px}div.footer-signature{display:block;align-items:center;padding:.5rem;text-align:center;font-size:.875rem;line-height:1.25rem}div.footer-signature div{margin:auto}div.footer-signature div span{--tw-text-opacity:1;color:rgb(156 163 175/var(--tw-text-opacity))}#tabs{margin-bottom:1rem;display:flex;border-bottom-width:1px;--tw-border-opacity:1;border-color:rgb(229 231 235/var(--tw-border-opacity))}#tabs:is(.dark *){--tw-border-opacity:1;border-color:rgb(55 65 81/var(--tw-border-opacity))}#tabs ul{margin-bottom:-1px;display:flex;flex-wrap:wrap;text-align:center;font-size:.875rem;line-height:1.25rem;font-weight:500}#tabs ul li button{display:inline-block;border-top-left-radius:.5rem;border-top-right-radius:.5rem;border-bottom-width:2px;padding:1rem}#tabs ul li button:hover{--tw-border-opacity:1;border-color:rgb(0 60 143/var(--tw-border-opacity));--tw-text-opacity:1;color:rgb(0 173 239/var(--tw-text-opacity))}#tabs ul li button:hover:is(.dark *){--tw-text-opacity:1;color:rgb(209 213 219/var(--tw-text-opacity))}#tabs ul li button.selected{--tw-border-opacity:1;border-color:rgb(0 60 143/var(--tw-border-opacity))}#tab-content{margin-bottom:1rem;display:block;min-width:100%;border-bottom-width:1px;--tw-border-opacity:1;border-color:rgb(229 231 235/var(--tw-border-opacity))}#tab-content:is(.dark *){--tw-border-opacity:1;border-color:rgb(55 65 81/var(--tw-border-opacity))}#tab-content div{display:block;min-width:100%;border-radius:.5rem;--tw-bg-opacity:1;background-color:rgb(249 250 251/var(--tw-bg-opacity));padding:1rem}#tab-content div:is(.dark *){--tw-bg-opacity:1;background-color:rgb(31 41 55/var(--tw-bg-opacity))}#tab-content div.hidden{display:none}ul.menu li.selected{background-color:#37b0f8}ul.menu li:hover{background-color:#c9cacf}*,:after,:before{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:rgba(59,130,246,.5);--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }::backdrop{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:rgba(59,130,246,.5);--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }
+:root{--primary:#0f2f3b;--header-bg:#1e3a8a;--breadcrumbs-bg:#0f2f3b;--link-fg:#37b0f8;--accent:#f5dd5d;--header-color:#ffc;--border-color:#ccc;--body-bg:#fff;--darkened-bg:#f8f8f8;--selected-bg:#e4e4e4;--body-quiet-color:#666;--hairline-color:#e8e8e8}*{font-family:Roboto,Helvetica,Arial,sans-serif;color:rgb(0 0 0/var(--tw-text-opacity))}*,.breadcrumbs{--tw-text-opacity:1}.breadcrumbs{padding-top:.5rem;padding-bottom:.5rem;color:rgb(0 60 143/var(--tw-text-opacity));//line-height:24px}ul.object-tools{margin-top:1.25rem;margin-bottom:1.25rem;margin-left:.75rem;display:flex;padding-top:.25rem;padding-bottom:.25rem;padding-right:1.25rem;justify-content:flex-end}ul.object-tools li{margin-left:.25rem;margin-right:.25rem;font-size:.875rem;line-height:1.25rem;font-weight:400}ul.object-tools li a{margin-top:.5rem;margin-bottom:.5rem;--tw-bg-opacity:1;background-color:rgb(0 60 143/var(--tw-bg-opacity));padding:.25rem .75rem;--tw-text-opacity:1;color:rgb(255 255 255/var(--tw-text-opacity));margin:0;color:#fff;cursor:pointer;font-size:.875rem;vertical-align:middle;justify-content:center;min-width:64px;box-shadow:0 3px 1px -2px rgba(0,0,0,.2),0 2px 2px 0 rgba(0,0,0,.14),0 1px 5px 0 rgba(0,0,0,.12);outline:0;border:0 initial;border-color:initial;-o-border-image:initial;border-image:initial;text-decoration:none;padding:6px 16px;border-radius:4px;transition:background-color .25s cubic-bezier(.4,0,.2,1),box-shadow .25s cubic-bezier(.4,0,.2,1),border-color .25s cubic-bezier(.4,0,.2,1),color .25s cubic-bezier(.4,0,.2,1);background-color:#003c8f}ul.object-tools li a.disabled{border-width:1px;--tw-border-opacity:1;border-color:rgb(0 60 143/var(--tw-border-opacity))}.admin-change-link{font-size:.75rem;line-height:1rem}.admin-change-link,.admin-change-link .icon{--tw-text-opacity:1;color:rgb(0 173 239/var(--tw-text-opacity))}.admin-change-link .icon{vertical-align:middle}.messages{position:absolute;top:2.5rem;z-index:50;display:flex;align-items:center;border-radius:.25rem;--tw-border-opacity:1;border-color:rgb(0 0 0/var(--tw-border-opacity));padding-left:.5rem;padding-right:.5rem;--tw-drop-shadow:drop-shadow(0 4px 3px rgba(0,0,0,.07)) drop-shadow(0 2px 2px rgba(0,0,0,.06));filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.messages .message{top:2.5rem;width:100%;border-radius:.25rem;--tw-bg-opacity:1;background-color:rgb(248 113 113/var(--tw-bg-opacity));padding:1rem;font-weight:700}.messages .message.success{--tw-bg-opacity:1;background-color:rgb(134 239 172/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(22 101 52/var(--tw-text-opacity))}.messages .message.error{--tw-bg-opacity:1;background-color:rgb(248 113 113/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(127 29 29/var(--tw-text-opacity))}.messages .message.warn,.messages .message.warning{--tw-bg-opacity:1;background-color:rgb(253 224 71/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(113 63 18/var(--tw-text-opacity))}.errors{--tw-bg-opacity:1;background-color:rgb(248 113 113/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(0 0 0/var(--tw-text-opacity))}#content #content-main,#content #content-related{display:flex;--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity));padding:.75rem}#content #content-related{margin-left:.25rem;margin-right:.25rem}table th{text-align:left;font-weight:400}table td{padding:.25rem}table tr:nth-child(2n).error,table tr:nth-child(odd).error{--tw-bg-opacity:1;background-color:rgb(254 202 202/var(--tw-bg-opacity))}table tr:nth-child(2n){--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}#login-form{align-content:center;border-radius:.25rem;--tw-bg-opacity:1;background-color:rgb(209 213 219/var(--tw-bg-opacity));padding:1.25rem}#login-form table{min-width:100%}#login-form table tr:nth-child(2n),#login-form table tr:nth-child(odd){--tw-bg-opacity:1;background-color:rgb(209 213 219/var(--tw-bg-opacity));text-align:center}#login-form table tr:nth-child(2n) td input,#login-form table tr:nth-child(odd) td input{min-width:100%}#changelist{display:flex;align-items:flex-start;justify-content:space-between}#changelist .changelist-form-container{width:50%}#changelist .changelist-form-container #changelist-form .results{width:300px;overflow:scroll}#changelist .changelist-form-container #changelist-form .actions{margin-bottom:.5rem}#changelist .changelist-form-container #changelist-form .actions select{padding-top:.25rem;padding-bottom:.25rem;font-size:.875rem;line-height:1.25rem}#changelist .changelist-form-container #changelist-form .actions button[type=submit]{font-size:.875rem;line-height:1.25rem}#changelist .changelist-form-container #changelist-form .actions .action-counter,#changelist .changelist-form-container #changelist-form .actions .all{margin-left:1.25rem}#changelist .changelist-form-container #changelist-form .actions .clear a,#changelist .changelist-form-container #changelist-form .actions .question a{--tw-text-opacity:1;color:rgb(0 60 143/var(--tw-text-opacity))}#changelist .clear{clear:both}#changelist table{border-right:none;border-collapse:collapse;border-color:var(--border-color)}#changelist table thead{display:table-header-group;vertical-align:middle;unicode-bidi:isolate}#changelist table thead th{//background:var(--selected-bg);font-size:.6875rem;border:none;border-bottom:1px solid var(--hairline-color);border-top:1px solid var(--hairline-color);color:var(--body-quiet-color);display:table-cell;font-weight:500;line-height:normal;padding:0;text-align:left;text-transform:uppercase;vertical-align:middle;white-space:nowrap}#changelist table thead th.sorted .text{margin-right:42px}#changelist table thead th .text span{padding:8px 10px}#changelist table thead th .text a{display:block;cursor:pointer;padding:8px 10px;color:var(--body-quiet-color);text-decoration:none;transition:color .15s,background .15s}#changelist table thead th .sortoptions{padding:9px 5px 0;float:right;display:block;text-align:right}#changelist table thead th .sortoptions a{position:relative;width:14px;height:14px;display:inline-block;background:url(sorting-icons.svg) 0 0 no-repeat;background-size:14px auto}#changelist table thead th .sortoptions a.ascending{background-position:0 -28px}#changelist table thead th .sortoptions a.descending{top:1px;background-position:0 -56px}#changelist table thead th .sortoptions a:link{transition:color .15s,background .15s}#changelist table thead th .sortoptions a.sortremove{background-position:0 0}#changelist table thead th.action-checkbox-column{width:1.5em;text-align:center}#changelist table tbody tr:hover{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}#changelist table tbody tr{border-width:1px;--tw-border-opacity:1;border-bottom-color:rgb(243 244 246/var(--tw-border-opacity))}#changelist table tbody td,#changelist table tbody th{padding:20px;white-space:nowrap;font-size:.8125rem;overflow:auto;cursor:pointer}#changelist table tbody td.action-checkbox,#changelist table tbody th.action-checkbox{width:1.5em;text-align:center}#changelist table tbody tr:nth-child(odd){--tw-border-opacity:1;border-bottom-color:rgb(75 85 99/var(--tw-border-opacity));--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity));padding-top:3rem;padding-bottom:3rem}#changelist table tbody tr:nth-child(odd):hover{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}#changelist table tbody tr:nth-child(2n){--tw-border-opacity:1;border-bottom-color:rgb(75 85 99/var(--tw-border-opacity));--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity));padding-top:3rem;padding-bottom:3rem}#changelist table tbody tr:nth-child(2n):hover{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}#changelist table tbody tr.selected:nth-child(2n),#changelist table tbody tr.selected:nth-child(odd){--tw-bg-opacity:1;background-color:rgb(219 234 254/var(--tw-bg-opacity))}#changelist-filter{margin-left:.5rem;border-radius:.25rem;border-width:1px;--tw-border-opacity:1;border-color:rgb(107 114 128/var(--tw-border-opacity))}#changelist-filter h2{font-size:.875rem;text-transform:uppercase;letter-spacing:.5px;padding:5px 15px;margin-bottom:12px;border-bottom:none}#changelist-filter #changelist-filter-extra-actions a{font-family:ui-sans-serif,system-ui,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji;font-size:.875rem;line-height:1.25rem;font-style:italic}#changelist-filter h3{font-weight:400;padding:0 15px;margin-bottom:10px}#changelist-filter details{margin-top:.25rem}#changelist-filter details>*{display:inline}#changelist-filter details summary{font-weight:400;padding:0 15px;margin-bottom:10px;list-style-type:none;cursor:pointer}#changelist-filter details summary:before{//content:"→";content:"►";//content:"▷";font-weight:bolder;font-size:18px;font-family:Courier New,Courier,monospace;color:var(--link-hover-color)}#changelist-filter details:last-child{margin-bottom:1rem}#changelist-filter details.active summary{font-weight:700;--tw-text-opacity:1;color:rgb(0 173 239/var(--tw-text-opacity))}#changelist-filter details[open]>summary:before{//content:"↓";content:"▼"}#changelist-filter ul{margin:0;padding-left:.5rem;padding-right:.5rem}#changelist-filter ul select{min-width:100%;padding:.25rem;font-size:.875rem;line-height:1.25rem;border-radius:4px;min-height:30px;height:28px}#changelist-filter ul select,#changelist-filter ul select:active{border-color:#ccc}#changelist-filter ul li{min-width:100%}#changelist-filter ul li .select2-container{border-radius:4px;min-width:100%}#changelist-filter ul li.selected{font-weight:700;--tw-text-opacity:1;color:rgb(0 173 239/var(--tw-text-opacity))}#changelist-filter ul:last-child{border-bottom:none}table thead th.action-checkbox-column{width:1.5em;text-align:center}.paginator{display:flex;border-width:1px;--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity));padding:.5rem}.paginator a,.paginator span{margin-left:.25rem;margin-right:.25rem;border-width:1px;--tw-border-opacity:1;border-color:rgb(75 85 99/var(--tw-border-opacity));padding-left:.5rem;padding-right:.5rem}.paginator span{--tw-bg-opacity:1;background-color:rgb(156 163 175/var(--tw-bg-opacity));--tw-text-opacity:1;color:rgb(255 255 255/var(--tw-text-opacity))}form input [type=date],form input[type=number],form input[type=password],form input[type=text],form select{border-radius:.25rem;border-width:1px;--tw-border-opacity:1;border-color:rgb(107 114 128/var(--tw-border-opacity));padding-top:.5rem;padding-bottom:.5rem;font-size:.75rem;line-height:1rem;font-weight:400}form .helptext{font-size:.75rem;line-height:1rem;font-style:italic}a.button,a.closelink,a.deletelink,button.button,input[type=button],input[type=submit]{margin:0;color:#fff;cursor:pointer;font-size:.875rem;vertical-align:middle;justify-content:center;min-width:64px;box-shadow:0 3px 1px -2px rgba(0,0,0,.2),0 2px 2px 0 rgba(0,0,0,.14),0 1px 5px 0 rgba(0,0,0,.12);outline:0;border:0 initial;border-color:initial;-o-border-image:initial;border-image:initial;text-decoration:none;line-height:inherit;height:33px;display:inline-block;padding:6px 16px;border-radius:4px;transition:background-color .25s cubic-bezier(.4,0,.2,1),box-shadow .25s cubic-bezier(.4,0,.2,1),border-color .25s cubic-bezier(.4,0,.2,1),color .25s cubic-bezier(.4,0,.2,1);background-color:#003c8f}a.button.default,a.closelink.default,a.deletelink.default,button.button.default,input[type=button].default,input[type=submit].default{background-color:#003c8f}a.button.closelink,a.closelink.closelink,a.deletelink.closelink,button.button.closelink,input[type=button].closelink,input[type=submit].closelink{background-color:rgba(99,92,15,.53)}.submit-row{margin-top:1.25rem}.submit-row a.closelink{@extend a.button;background-color:rgba(99,92,15,.53)}.submit-row a.deletelink{background-color:#cf2222;float:right}fieldset.module{border:none;margin:0 0 30px;background:var(--body-bg);min-width:0;padding:0}fieldset.module .fieldBox{margin-right:20px}fieldset.module .form-row{overflow:hidden;padding:10px;font-size:.8125rem;border-bottom:1px solid var(--hairline-color);margin-bottom:0}fieldset.module .form-row input,fieldset.module .form-row select{padding-top:.25rem;padding-bottom:.25rem;font-size:.875rem;line-height:1.25rem}fieldset.module .form-row .flex-container{display:flex}fieldset.module .form-row .flex-container .related-widget-wrapper{display:flex;gap:0 10px;flex-grow:1;flex-wrap:wrap;margin-bottom:5px}fieldset.module .form-row label{display:block;padding:4px 10px 0 0;min-width:160px;width:160px;word-wrap:break-word;color:var(--body-quiet-color);font-size:.8125rem}.beneficiary-form input[type=text],.beneficiary-form select{min-width:100%}.beneficiary-form .validation-error{font-style:italic}body.web #select2-id_program-results>li.select2-results__option.select2-results__option--highlighted,body.web li.select2-results__option.select2-results__option--highlighted{--tw-bg-opacity:1;background-color:rgb(0 173 239/var(--tw-bg-opacity))}body.web header{--tw-bg-opacity:1;background-color:rgb(35 57 68/var(--tw-bg-opacity));color:#fff}body.web header .select2{background-color:var(--body-bg)}body.web header .select2-container.select2-container--default{font-size:.875rem;line-height:1.25rem;min-width:600px;height:38px;border-radius:4px;border-width:0}body.web header .select2-container.select2-container--default .select2-results__option--highlighted[aria-selected]{--tw-bg-opacity:1;background-color:rgb(0 173 239/var(--tw-bg-opacity));color:#fff}body.web header .select2-container.select2-container--default .selection{font-size:.875rem;line-height:1.25rem;height:34px;border-radius:4px;line-height:34px;border-width:0}body.web header .select2-container.select2-container--default .selection .select2-selection.select2-selection--single{height:34px;border-radius:4px;border-width:0}body.web header .select2-container.select2-container--default .selection .select2-selection.select2-selection--single .select2-selection__rendered{line-height:38px;border-color:rgb(107 114 128/var(--tw-border-opacity));border-width:1px;border-radius:4px}body.web header .select2-container.select2-container--default .selection .select2-selection.select2-selection--single .select2-selection__arrow{height:38px;line-height:38px}body.web select{min-width:12rem;padding-top:.5rem;padding-bottom:.5rem;font-size:.875rem;line-height:1.25rem;border-radius:4px;border-width:1px;border-color:rgb(107 114 128/var(--tw-border-opacity));padding-right:2.5rem}body.web svg.sidebar-button{fill:#fff;width:1em;height:1em;display:inline-block;font-size:1.5rem;flex-shrink:0;-webkit-user-select:none;-moz-user-select:none;user-select:none;color:#fff;border-color:#fff}.title-top{box-shadow:0 2px 4px -1px rgba(0,0,0,.2),0 4px 5px 0 rgba(0,0,0,.14),0 1px 10px 0 rgba(0,0,0,.12);position:relative;width:100%;background-color:#fff}.title-top h2{font-size:1.5rem;font-weight:400;line-height:1.334;letter-spacing:0;color:rgba(0,0,0,.87);padding:28px 44px}div.footer-signature{display:block;align-items:center;padding:.5rem;text-align:center;font-size:.875rem;line-height:1.25rem}div.footer-signature div{margin:auto}div.footer-signature div span{--tw-text-opacity:1;color:rgb(156 163 175/var(--tw-text-opacity))}#tabs{margin-bottom:1rem;display:flex;border-bottom-width:1px;--tw-border-opacity:1;border-color:rgb(229 231 235/var(--tw-border-opacity))}#tabs:is(.dark *){--tw-border-opacity:1;border-color:rgb(55 65 81/var(--tw-border-opacity))}#tabs ul{margin-bottom:-1px;display:flex;flex-wrap:wrap;text-align:center;font-size:.875rem;line-height:1.25rem;font-weight:500}#tabs ul li button{display:inline-block;border-top-left-radius:.5rem;border-top-right-radius:.5rem;border-bottom-width:2px;padding:1rem}#tabs ul li button:hover{--tw-border-opacity:1;border-color:rgb(0 60 143/var(--tw-border-opacity));--tw-text-opacity:1;color:rgb(0 173 239/var(--tw-text-opacity))}#tabs ul li button:hover:is(.dark *){--tw-text-opacity:1;color:rgb(209 213 219/var(--tw-text-opacity))}#tabs ul li button.selected{--tw-border-opacity:1;border-color:rgb(0 60 143/var(--tw-border-opacity))}#tab-content{margin-bottom:1rem;display:block;min-width:100%;border-bottom-width:1px;--tw-border-opacity:1;border-color:rgb(229 231 235/var(--tw-border-opacity))}#tab-content:is(.dark *){--tw-border-opacity:1;border-color:rgb(55 65 81/var(--tw-border-opacity))}#tab-content div{display:block;min-width:100%;border-radius:.5rem;--tw-bg-opacity:1;background-color:rgb(249 250 251/var(--tw-bg-opacity));padding:1rem}#tab-content div:is(.dark *){--tw-bg-opacity:1;background-color:rgb(31 41 55/var(--tw-bg-opacity))}#tab-content div.hidden{display:none}ul.menu li.selected{background-color:#37b0f8}ul.menu li:hover{background-color:#c9cacf}*,:after,:before{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:rgba(59,130,246,.5);--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }::backdrop{--tw-border-spacing-x:0;--tw-border-spacing-y:0;--tw-translate-x:0;--tw-translate-y:0;--tw-rotate:0;--tw-skew-x:0;--tw-skew-y:0;--tw-scale-x:1;--tw-scale-y:1;--tw-pan-x: ;--tw-pan-y: ;--tw-pinch-zoom: ;--tw-scroll-snap-strictness:proximity;--tw-gradient-from-position: ;--tw-gradient-via-position: ;--tw-gradient-to-position: ;--tw-ordinal: ;--tw-slashed-zero: ;--tw-numeric-figure: ;--tw-numeric-spacing: ;--tw-numeric-fraction: ;--tw-ring-inset: ;--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:rgba(59,130,246,.5);--tw-ring-offset-shadow:0 0 #0000;--tw-ring-shadow:0 0 #0000;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;--tw-blur: ;--tw-brightness: ;--tw-contrast: ;--tw-grayscale: ;--tw-hue-rotate: ;--tw-invert: ;--tw-saturate: ;--tw-sepia: ;--tw-drop-shadow: ;--tw-backdrop-blur: ;--tw-backdrop-brightness: ;--tw-backdrop-contrast: ;--tw-backdrop-grayscale: ;--tw-backdrop-hue-rotate: ;--tw-backdrop-invert: ;--tw-backdrop-opacity: ;--tw-backdrop-saturate: ;--tw-backdrop-sepia: ;--tw-contain-size: ;--tw-contain-layout: ;--tw-contain-paint: ;--tw-contain-style: }
/*! tailwindcss v3.4.14 | MIT License | https://tailwindcss.com*/*,:after,:before{box-sizing:border-box;border:0 solid #e5e7eb}:after,:before{--tw-content:""}:host,html{line-height:1.5;-webkit-text-size-adjust:100%;-moz-tab-size:4;-o-tab-size:4;tab-size:4;font-family:ui-sans-serif,system-ui,sans-serif,Apple Color Emoji,Segoe UI Emoji,Segoe UI Symbol,Noto Color Emoji;font-feature-settings:normal;font-variation-settings:normal;-webkit-tap-highlight-color:transparent}body{margin:0;line-height:inherit}hr{height:0;color:inherit;border-top-width:1px}abbr:where([title]){-webkit-text-decoration:underline dotted;text-decoration:underline dotted}h1,h2,h3,h4,h5,h6{font-size:inherit;font-weight:inherit}a{color:inherit;text-decoration:inherit}b,strong{font-weight:bolder}code,kbd,pre,samp{font-family:ui-monospace,SFMono-Regular,Menlo,Monaco,Consolas,Liberation Mono,Courier New,monospace;font-feature-settings:normal;font-variation-settings:normal;font-size:1em}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}table{text-indent:0;border-color:inherit;border-collapse:collapse}button,input,optgroup,select,textarea{font-family:inherit;font-feature-settings:inherit;font-variation-settings:inherit;font-size:100%;font-weight:inherit;line-height:inherit;letter-spacing:inherit;color:inherit;margin:0;padding:0}button,select{text-transform:none}button,input:where([type=button]),input:where([type=reset]),input:where([type=submit]){-webkit-appearance:button;background-color:transparent;background-image:none}:-moz-focusring{outline:auto}:-moz-ui-invalid{box-shadow:none}progress{vertical-align:baseline}::-webkit-inner-spin-button,::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}summary{display:list-item}blockquote,dd,dl,figure,h1,h2,h3,h4,h5,h6,hr,p,pre{margin:0}fieldset{margin:0}fieldset,legend{padding:0}menu,ol,ul{list-style:none;margin:0;padding:0}dialog{padding:0}textarea{resize:vertical}input::-moz-placeholder,textarea::-moz-placeholder{opacity:1;color:#9ca3af}input::placeholder,textarea::placeholder{opacity:1;color:#9ca3af}[role=button],button{cursor:pointer}:disabled{cursor:default}audio,canvas,embed,iframe,img,object,svg,video{display:block;vertical-align:middle}img,video{max-width:100%;height:auto}[hidden]:where(:not([hidden=until-found])){display:none}[multiple],[type=date],[type=datetime-local],[type=email],[type=month],[type=number],[type=password],[type=search],[type=tel],[type=text],[type=time],[type=url],[type=week],input:where(:not([type])),select,textarea{-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:#fff;border-color:#6b7280;border-width:1px;border-radius:0;padding:.5rem .75rem;font-size:1rem;line-height:1.5rem;--tw-shadow:0 0 #0000}[multiple]:focus,[type=date]:focus,[type=datetime-local]:focus,[type=email]:focus,[type=month]:focus,[type=number]:focus,[type=password]:focus,[type=search]:focus,[type=tel]:focus,[type=text]:focus,[type=time]:focus,[type=url]:focus,[type=week]:focus,input:where(:not([type])):focus,select:focus,textarea:focus{outline:2px solid transparent;outline-offset:2px;--tw-ring-inset:var(--tw-empty,/*!*/ /*!*/);--tw-ring-offset-width:0px;--tw-ring-offset-color:#fff;--tw-ring-color:#2563eb;--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(1px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow);border-color:#2563eb}input::-moz-placeholder,textarea::-moz-placeholder{color:#6b7280;opacity:1}input::placeholder,textarea::placeholder{color:#6b7280;opacity:1}::-webkit-datetime-edit-fields-wrapper{padding:0}::-webkit-date-and-time-value{min-height:1.5em;text-align:inherit}::-webkit-datetime-edit{display:inline-flex}::-webkit-datetime-edit,::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-meridiem-field,::-webkit-datetime-edit-millisecond-field,::-webkit-datetime-edit-minute-field,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-second-field,::-webkit-datetime-edit-year-field{padding-top:0;padding-bottom:0}select{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 20 20'%3E%3Cpath stroke='%236b7280' stroke-linecap='round' stroke-linejoin='round' stroke-width='1.5' d='m6 8 4 4 4-4'/%3E%3C/svg%3E");background-position:right .5rem center;background-repeat:no-repeat;background-size:1.5em 1.5em;padding-right:2.5rem;-webkit-print-color-adjust:exact;print-color-adjust:exact}[multiple],[size]:where(select:not([size="1"])){background-image:none;background-position:0 0;background-repeat:unset;background-size:initial;padding-right:.75rem;-webkit-print-color-adjust:unset;print-color-adjust:unset}[type=checkbox],[type=radio]{-webkit-appearance:none;-moz-appearance:none;appearance:none;padding:0;-webkit-print-color-adjust:exact;print-color-adjust:exact;display:inline-block;vertical-align:middle;background-origin:border-box;-webkit-user-select:none;-moz-user-select:none;user-select:none;flex-shrink:0;height:1rem;width:1rem;color:#2563eb;background-color:#fff;border-color:#6b7280;border-width:1px;--tw-shadow:0 0 #0000}[type=checkbox]{border-radius:0}[type=radio]{border-radius:100%}[type=checkbox]:focus,[type=radio]:focus{outline:2px solid transparent;outline-offset:2px;--tw-ring-inset:var(--tw-empty,/*!*/ /*!*/);--tw-ring-offset-width:2px;--tw-ring-offset-color:#fff;--tw-ring-color:#2563eb;--tw-ring-offset-shadow:var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);--tw-ring-shadow:var(--tw-ring-inset) 0 0 0 calc(2px + var(--tw-ring-offset-width)) var(--tw-ring-color);box-shadow:var(--tw-ring-offset-shadow),var(--tw-ring-shadow),var(--tw-shadow)}[type=checkbox]:checked,[type=radio]:checked{border-color:transparent;background-color:currentColor;background-size:100% 100%;background-position:50%;background-repeat:no-repeat}[type=checkbox]:checked{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' viewBox='0 0 16 16'%3E%3Cpath d='M12.207 4.793a1 1 0 0 1 0 1.414l-5 5a1 1 0 0 1-1.414 0l-2-2a1 1 0 0 1 1.414-1.414L6.5 9.086l4.293-4.293a1 1 0 0 1 1.414 0'/%3E%3C/svg%3E")}@media (forced-colors:active) {[type=checkbox]:checked{-webkit-appearance:auto;-moz-appearance:auto;appearance:auto}}[type=radio]:checked{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' viewBox='0 0 16 16'%3E%3Ccircle cx='8' cy='8' r='3'/%3E%3C/svg%3E")}@media (forced-colors:active) {[type=radio]:checked{-webkit-appearance:auto;-moz-appearance:auto;appearance:auto}}[type=checkbox]:checked:focus,[type=checkbox]:checked:hover,[type=radio]:checked:focus,[type=radio]:checked:hover{border-color:transparent;background-color:currentColor}[type=checkbox]:indeterminate{background-image:url("data:image/svg+xml;charset=utf-8,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='none' viewBox='0 0 16 16'%3E%3Cpath stroke='%23fff' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='M4 8h8'/%3E%3C/svg%3E");border-color:transparent;background-color:currentColor;background-size:100% 100%;background-position:50%;background-repeat:no-repeat}@media (forced-colors:active) {[type=checkbox]:indeterminate{-webkit-appearance:auto;-moz-appearance:auto;appearance:auto}}[type=checkbox]:indeterminate:focus,[type=checkbox]:indeterminate:hover{border-color:transparent;background-color:currentColor}[type=file]{background:unset;border-color:inherit;border-width:0;border-radius:0;padding:0;font-size:unset;line-height:inherit}[type=file]:focus{outline:1px solid ButtonText;outline:1px auto -webkit-focus-ring-color}.static{position:static}.float-right{float:right}.m-2{margin:.5rem}.m-5{margin:1.25rem}.m-auto{margin:auto}.my-5{margin-top:1.25rem}.mb-5,.my-5{margin-bottom:1.25rem}.me-2{margin-inline-end:.5rem}.mt-10{margin-top:2.5rem}.mt-3{margin-top:.75rem}.mt-40{margin-top:10rem}.mt-5{margin-top:1.25rem}.block{display:block}.inline{display:inline}.flex{display:flex}.inline-flex{display:inline-flex}.table{display:table}.contents{display:contents}.hidden{display:none}.h-12{height:3rem}.h-16{height:4rem}.h-5\/6{height:83.333333%}.h-screen{height:100vh}.w-1\/4{width:25%}.w-1\/6{width:16.666667%}.w-16{width:4rem}.w-52{width:13rem}.w-full{width:100%}.min-w-full{min-width:100%}.flex-1{flex:1 1 0%}.flex-none{flex:none}.flex-shrink{flex-shrink:1}.flex-grow,.grow{flex-grow:1}.table-fixed{table-layout:fixed}.caption-bottom{caption-side:bottom}.border-collapse{border-collapse:collapse}.cursor-pointer{cursor:pointer}.flex-col{flex-direction:column}.items-center{align-items:center}.justify-center{justify-content:center}.overflow-y-auto{overflow-y:auto}.rounded{border-radius:.25rem}.border{border-width:1px}.border-2{border-width:2px}.border-gray-300{--tw-border-opacity:1;border-color:rgb(209 213 219/var(--tw-border-opacity))}.border-t-black{--tw-border-opacity:1;border-top-color:rgb(0 0 0/var(--tw-border-opacity))}.bg-gray-100{--tw-bg-opacity:1;background-color:rgb(243 244 246/var(--tw-bg-opacity))}.bg-gray-200{--tw-bg-opacity:1;background-color:rgb(229 231 235/var(--tw-bg-opacity))}.bg-unicef-blue{--tw-bg-opacity:1;background-color:rgb(0 173 239/var(--tw-bg-opacity))}.bg-unicef-gray{--tw-bg-opacity:1;background-color:rgb(35 57 68/var(--tw-bg-opacity))}.bg-white{--tw-bg-opacity:1;background-color:rgb(255 255 255/var(--tw-bg-opacity))}.p-0{padding:0}.p-1{padding:.25rem}.p-2{padding:.5rem}.p-5{padding:1.25rem}.px-10{padding-left:2.5rem;padding-right:2.5rem}.px-4{padding-left:1rem;padding-right:1rem}.px-5{padding-left:1.25rem;padding-right:1.25rem}.py-2{padding-top:.5rem;padding-bottom:.5rem}.py-5{padding-top:1.25rem;padding-bottom:1.25rem}.pl-5{padding-left:1.25rem}.pl-8{padding-left:2rem}.pr-10{padding-right:2.5rem}.pr-2{padding-right:.5rem}.pr-4{padding-right:1rem}.pt-1{padding-top:.25rem}.pt-10{padding-top:2.5rem}.pt-2{padding-top:.5rem}.pt-20{padding-top:5rem}.pt-3{padding-top:.75rem}.text-center{text-align:center}.align-middle{vertical-align:middle}.text-2xl{font-size:1.5rem;line-height:2rem}.text-5xl{font-size:3rem;line-height:1}.text-9xl{font-size:8rem;line-height:1}.text-xl{font-size:1.25rem;line-height:1.75rem}.font-bold{font-weight:700}.font-extralight{font-weight:200}.text-blue-700{--tw-text-opacity:1;color:rgb(29 78 216/var(--tw-text-opacity))}.text-gray-400{--tw-text-opacity:1;color:rgb(156 163 175/var(--tw-text-opacity))}.text-white{--tw-text-opacity:1;color:rgb(255 255 255/var(--tw-text-opacity))}.drop-shadow-md{--tw-drop-shadow:drop-shadow(0 4px 3px rgba(0,0,0,.07)) drop-shadow(0 2px 2px rgba(0,0,0,.06))}.drop-shadow-md,.filter{filter:var(--tw-blur) var(--tw-brightness) var(--tw-contrast) var(--tw-grayscale) var(--tw-hue-rotate) var(--tw-invert) var(--tw-saturate) var(--tw-sepia) var(--tw-drop-shadow)}.focus\:outline-none:focus{outline:2px solid transparent;outline-offset:2px}
diff --git a/src/country_workspace/workspaces/theme/static_src/src/_common.scss b/src/country_workspace/workspaces/theme/static_src/src/_common.scss
index 44ba6b78..2aad08f0 100644
--- a/src/country_workspace/workspaces/theme/static_src/src/_common.scss
+++ b/src/country_workspace/workspaces/theme/static_src/src/_common.scss
@@ -86,7 +86,9 @@ ul.object-tools {
}
}
}
-
+.errors{
+ @apply bg-red-400 text-black;
+}
#content {
#content-main {
@apply p-3 bg-white flex;
diff --git a/tests/admin/test_admin_cleaners.py b/tests/admin/test_admin_cleaners.py
new file mode 100644
index 00000000..4711bec4
--- /dev/null
+++ b/tests/admin/test_admin_cleaners.py
@@ -0,0 +1,22 @@
+from unittest import mock
+from unittest.mock import Mock
+
+import pytest
+
+from country_workspace.workspaces.admin.cleaners.validate import validate_program
+
+
+@pytest.fixture
+def household():
+ from testutils.factories import CountryHouseholdFactory, ProgramFactory
+
+ program = ProgramFactory()
+ return CountryHouseholdFactory(batch__program=program, batch__country_office=program.country_office)
+
+
+def test_validate_program(household):
+ assert validate_program(Mock(program=household.program)) == {"invalid": 0, "total": 0, "valid": 1}
+ with mock.patch(
+ "country_workspace.workspaces.admin.cleaners.validate.Household.validate_with_checker", return_value=False
+ ):
+ assert validate_program(Mock(program=household.program)) == {"invalid": 1, "total": 0, "valid": 0}
diff --git a/tests/admin/test_admin_sync.py b/tests/admin/test_admin_sync.py
index 208bc99f..dac9cf67 100644
--- a/tests/admin/test_admin_sync.py
+++ b/tests/admin/test_admin_sync.py
@@ -1,11 +1,13 @@
from typing import TYPE_CHECKING
+
import pytest
-from pytest_mock import MockerFixture
from django.urls import reverse
+from pytest_mock import MockerFixture
if TYPE_CHECKING:
from django_webtest.pytest_plugin import MixinWithInstanceVariables
from testutils.types import CWTestApp
+
from country_workspace.models import User
diff --git a/tests/contrib/aurora/stub.py b/tests/contrib/aurora/stub.py
index 8c5071cc..856b942b 100644
--- a/tests/contrib/aurora/stub.py
+++ b/tests/contrib/aurora/stub.py
@@ -1,5 +1,4 @@
-from typing import Final, Any
-
+from typing import Any, Final
project: Final[dict[str, dict[str, list[dict[str, Any]]]]] = {
"results": [
diff --git a/tests/contrib/aurora/test_aurora_client.py b/tests/contrib/aurora/test_aurora_client.py
index c641f444..9866dc61 100644
--- a/tests/contrib/aurora/test_aurora_client.py
+++ b/tests/contrib/aurora/test_aurora_client.py
@@ -1,9 +1,10 @@
-import pytest
-import responses
-import requests
import re
from typing import Callable
+import pytest
+import requests
+import responses
+
from country_workspace.contrib.aurora.client import AuroraClient
from country_workspace.exceptions import RemoteError
diff --git a/tests/contrib/aurora/test_aurora_sync.py b/tests/contrib/aurora/test_aurora_sync.py
index 105c0e83..8fedfae3 100644
--- a/tests/contrib/aurora/test_aurora_sync.py
+++ b/tests/contrib/aurora/test_aurora_sync.py
@@ -1,23 +1,21 @@
-import pytest
-import responses
import re
-
-from typing import Generator
-
from contextlib import nullcontext
+from typing import Generator
+import pytest
+import responses
+from constance import config
from django.core.cache import cache
from pytest_mock import MockerFixture
+
+from country_workspace.contrib.aurora.client import AuroraClient
+from country_workspace.contrib.aurora.models import Project
from country_workspace.contrib.aurora.sync import (
sync_all,
sync_projects,
sync_registrations,
)
-from country_workspace.contrib.aurora.client import AuroraClient
-from country_workspace.contrib.aurora.models import Project
from country_workspace.models import SyncLog
-from constance import config
-
from tests.contrib.aurora import stub
diff --git a/tests/contrib/kobo/api/client/test_client_auth.py b/tests/contrib/kobo/api/client/test_client_auth.py
index 3946bff1..230e3586 100644
--- a/tests/contrib/kobo/api/client/test_client_auth.py
+++ b/tests/contrib/kobo/api/client/test_client_auth.py
@@ -1,8 +1,9 @@
from unittest.mock import Mock
-from country_workspace.contrib.kobo.api.client.auth import Auth, TOKEN, AUTHORIZATION
from requests.models import PreparedRequest
+from country_workspace.contrib.kobo.api.client.auth import AUTHORIZATION, TOKEN, Auth
+
def test_auth_call():
request = Mock(spec=PreparedRequest)
diff --git a/tests/contrib/kobo/api/client/test_client_helpers.py b/tests/contrib/kobo/api/client/test_client_helpers.py
index dd9aff2c..1b569adb 100644
--- a/tests/contrib/kobo/api/client/test_client_helpers.py
+++ b/tests/contrib/kobo/api/client/test_client_helpers.py
@@ -2,18 +2,19 @@
from unittest.mock import Mock, call
import pytest
-from country_workspace.contrib.kobo.api.data.helpers import download_attachments
-from country_workspace.contrib.kobo.api.raw.asset_list import Asset
from pytest_mock import MockerFixture
+
from country_workspace.contrib.kobo.api.client.helpers import (
+ get_asset,
+ get_asset_list,
get_asset_list_url,
- handle_paginated_response,
get_raw_asset_list,
get_raw_submission_list,
- get_asset_list,
get_submission_list,
- get_asset,
+ handle_paginated_response,
)
+from country_workspace.contrib.kobo.api.data.helpers import download_attachments
+from country_workspace.contrib.kobo.api.raw.asset_list import Asset
from country_workspace.contrib.kobo.api.raw.common import ListResponse
BASE_URL = "https://test.org"
diff --git a/tests/contrib/kobo/api/client/test_client_main.py b/tests/contrib/kobo/api/client/test_client_main.py
index fa8d1ab1..7f2835f0 100644
--- a/tests/contrib/kobo/api/client/test_client_main.py
+++ b/tests/contrib/kobo/api/client/test_client_main.py
@@ -1,6 +1,7 @@
-from country_workspace.contrib.kobo.api.client.main import Client, ACCEPT_JSON_HEADERS
from pytest_mock import MockerFixture
+from country_workspace.contrib.kobo.api.client.main import ACCEPT_JSON_HEADERS, Client
+
def test_client(mocker: MockerFixture) -> None:
session_class = mocker.patch("country_workspace.contrib.kobo.api.client.main.Session")
diff --git a/tests/contrib/kobo/api/data/test_data_helpers.py b/tests/contrib/kobo/api/data/test_data_helpers.py
index 7b00a985..6b241d5d 100644
--- a/tests/contrib/kobo/api/data/test_data_helpers.py
+++ b/tests/contrib/kobo/api/data/test_data_helpers.py
@@ -1,10 +1,10 @@
from typing import cast
-from unittest.mock import Mock, MagicMock
+from unittest.mock import MagicMock, Mock
import pytest
from pytest_mock import MockerFixture
-from country_workspace.contrib.kobo.api.data.helpers import filter_out_meta_data, download_attachments, VALUE_FORMAT
+from country_workspace.contrib.kobo.api.data.helpers import VALUE_FORMAT, download_attachments, filter_out_meta_data
from country_workspace.contrib.kobo.api.data.submission import Submission
diff --git a/tests/contrib/kobo/test_kobo_sync.py b/tests/contrib/kobo/test_kobo_sync.py
index f17aa5bb..81e2f507 100644
--- a/tests/contrib/kobo/test_kobo_sync.py
+++ b/tests/contrib/kobo/test_kobo_sync.py
@@ -3,18 +3,19 @@
import pytest
from constance.test.unittest import override_config
+from pytest_mock import MockerFixture
+
from country_workspace.contrib.kobo.api.data.submission import Submission
from country_workspace.contrib.kobo.sync import (
- make_client,
- extract_household_data,
- create_individuals,
- create_household,
- import_data,
+ ASSET_CACHE_KEY,
ImportResult,
+ create_household,
+ create_individuals,
+ extract_household_data,
import_asset,
- ASSET_CACHE_KEY,
+ import_data,
+ make_client,
)
-from pytest_mock import MockerFixture
EMPTY = ""
TOKEN = "token"
diff --git a/tests/extras/testutils/factories/__init__.py b/tests/extras/testutils/factories/__init__.py
index 1ee1a589..706d15fc 100644
--- a/tests/extras/testutils/factories/__init__.py
+++ b/tests/extras/testutils/factories/__init__.py
@@ -5,6 +5,7 @@
from factory.django import DjangoModelFactory
from pytest_factoryboy import register
+from .aurora import ProjectFactory, RegistrationFactory # noqa
from .base import AutoRegisterModelFactory, TAutoRegisterModelFactory, factories_registry
from .batch import BatchFactory, CountryBatchFactory # noqa
from .django_celery_beat import PeriodicTaskFactory # noqa
@@ -14,7 +15,6 @@
from .locations import AreaFactory, AreaTypeFactory, CountryFactory # noqa
from .office import OfficeFactory # noqa
from .program import CountryProgramFactory, ProgramFactory # noqa
-from .aurora import ProjectFactory, RegistrationFactory # noqa
from .smart_fields import DataCheckerFactory, FieldDefinitionFactory, FieldsetFactory, FlexFieldFactory # noqa
from .social import SocialAuthUserFactory # noqa
from .sync import SyncLogFactory # noqa
diff --git a/tests/extras/testutils/factories/aurora.py b/tests/extras/testutils/factories/aurora.py
index 2555a7e8..0d2a83d7 100644
--- a/tests/extras/testutils/factories/aurora.py
+++ b/tests/extras/testutils/factories/aurora.py
@@ -2,10 +2,12 @@
import factory.fuzzy
from country_workspace.contrib.aurora.models import Project, Registration
+
+from .base import AutoRegisterModelFactory
from .program import ProgramFactory
-class ProjectFactory(factory.django.DjangoModelFactory):
+class ProjectFactory(AutoRegisterModelFactory):
name = factory.Sequence(lambda n: f"Project {n}")
reference_pk = factory.fuzzy.FuzzyInteger(100)
program = factory.SubFactory(ProgramFactory)
@@ -15,7 +17,7 @@ class Meta:
django_get_or_create = ("name",)
-class RegistrationFactory(factory.django.DjangoModelFactory):
+class RegistrationFactory(AutoRegisterModelFactory):
name = factory.Sequence(lambda n: f"Registration {n}")
active = factory.fuzzy.FuzzyChoice([True, False])
reference_pk = factory.fuzzy.FuzzyInteger(100)
diff --git a/tests/extras/testutils/factories/smart_fields.py b/tests/extras/testutils/factories/smart_fields.py
index f7acd84e..229976d1 100644
--- a/tests/extras/testutils/factories/smart_fields.py
+++ b/tests/extras/testutils/factories/smart_fields.py
@@ -2,7 +2,7 @@
from django import forms
from hope_flex_fields.models import DataChecker, DataCheckerFieldset, FieldDefinition, Fieldset, FlexField
-from testutils.factories import AutoRegisterModelFactory
+from .base import AutoRegisterModelFactory
class FieldDefinitionFactory(AutoRegisterModelFactory):
@@ -59,7 +59,12 @@ def fields(self, create, extracted, **kwargs):
if extracted:
fs = FieldsetFactory()
for i in extracted:
- FlexFieldFactory(fieldset=fs, name=i)
+ if isinstance(i, str):
+ FlexFieldFactory(fieldset=fs, name=i)
+ elif isinstance(i, (tuple, list)):
+ df = FieldDefinitionFactory(field_type=i[1])
+ FlexFieldFactory(fieldset=fs, name=i[0], definition=df)
+
self.fieldsets.add(fs)
diff --git a/tests/models/test_m_household.py b/tests/models/test_m_household.py
index 48c0f976..2c35192c 100644
--- a/tests/models/test_m_household.py
+++ b/tests/models/test_m_household.py
@@ -1,9 +1,13 @@
from typing import TYPE_CHECKING
+from unittest import mock
+from unittest.mock import Mock
import pytest
+
if TYPE_CHECKING:
- from country_workspace.workspaces.models import CountryHousehold
+ from country_workspace.workspaces.models import CountryHousehold, CountryIndividual
+ from country_workspace.models import Household
@pytest.fixture
@@ -13,6 +17,22 @@ def household() -> "CountryHousehold":
return CountryHouseholdFactory()
+@pytest.fixture
+def individual(household) -> "CountryIndividual":
+ from testutils.factories import CountryIndividualFactory
+
+ return CountryIndividualFactory(household=household)
+
+
def test_properties(household: "CountryHousehold"):
assert household.program == household.batch.program
assert household.country_office == household.batch.country_office
+
+
+def test_validate_with_checker(individual: "CountryHousehold"):
+ household: Household = individual.household
+ assert household.validate_with_checker()
+ assert household.errors == {}
+ with mock.patch.object(household.program.beneficiary_validator, "validate", Mock(return_value=["Error"])):
+ assert not household.validate_with_checker()
+ assert household.errors == {"dct": ["Error"]}
diff --git a/tests/test_celery.py b/tests/test_celery.py
index eb0e7a15..14e0860a 100644
--- a/tests/test_celery.py
+++ b/tests/test_celery.py
@@ -1,4 +1,5 @@
from country_workspace.config.celery import app, init_sentry
+from country_workspace.tasks import removed_expired_jobs
def test_celery_app(**kwargs):
@@ -9,3 +10,7 @@ def test_celery_app(**kwargs):
def test_celery_init_sentry(**kwargs):
init_sentry()
assert True
+
+
+def test_removed_expired_jobs(**kwargs):
+ removed_expired_jobs()
diff --git a/tests/test_sync.py b/tests/test_sync.py
index 2062c059..376d02bc 100644
--- a/tests/test_sync.py
+++ b/tests/test_sync.py
@@ -1,5 +1,6 @@
-import pytest
from io import StringIO
+
+import pytest
from django import forms
from country_workspace.contrib.hope.sync.office import sync_all, sync_offices, sync_programs
diff --git a/tests/utils/tests_utils_fields.py b/tests/utils/test_utils_fields.py
similarity index 100%
rename from tests/utils/tests_utils_fields.py
rename to tests/utils/test_utils_fields.py
diff --git a/tests/workspace/actions/stub.py b/tests/workspace/actions/stub.py
index 72d08598..2195815b 100644
--- a/tests/workspace/actions/stub.py
+++ b/tests/workspace/actions/stub.py
@@ -1,6 +1,5 @@
from typing import Final
-
header_base: Final[tuple[str, ...]] = ["id", "version"]
header_add: Final[dict[str, tuple[str, ...]]] = {
"hh": ["admin1", "admin2"],
diff --git a/tests/workspace/actions/test_ws_bulk.py b/tests/workspace/actions/test_ws_bulk.py
index 9b886063..ea2de38e 100644
--- a/tests/workspace/actions/test_ws_bulk.py
+++ b/tests/workspace/actions/test_ws_bulk.py
@@ -1,6 +1,7 @@
import io
import os
from typing import TYPE_CHECKING, Any
+from unittest import mock
import openpyxl
import pytest
@@ -8,14 +9,11 @@
from django.urls import reverse
from testutils.factories import FlexFieldFactory
from testutils.utils import select_office
-from unittest import mock
from country_workspace.state import state
from country_workspace.workspaces.admin.cleaners.bulk_update import TYPES, create_xls_importer
-
from tests.workspace.actions import stub
-
if TYPE_CHECKING:
from django_webtest import DjangoTestApp
from django_webtest.pytest_plugin import MixinWithInstanceVariables
diff --git a/tests/workspace/cleaners/test_bulk_update.py b/tests/workspace/cleaners/test_bulk_update.py
index 541a68ef..a81b712d 100644
--- a/tests/workspace/cleaners/test_bulk_update.py
+++ b/tests/workspace/cleaners/test_bulk_update.py
@@ -1,14 +1,15 @@
import io
+from typing import TYPE_CHECKING
+
import pytest
-from pytest_mock import MockerFixture
from constance.test import override_config
-from typing import TYPE_CHECKING
+from pytest_mock import MockerFixture
-from country_workspace.workspaces.admin.cleaners.bulk_update import bulk_update_household
from country_workspace.models import AsyncJob
+from country_workspace.workspaces.admin.cleaners.bulk_update import bulk_update_household
if TYPE_CHECKING:
- from country_workspace.workspaces.models import CountryProgram, CountryHousehold
+ from country_workspace.workspaces.models import CountryHousehold, CountryProgram
@pytest.fixture
diff --git a/tests/workspace/test_ws_household.py b/tests/workspace/test_ws_household.py
index 85c301a6..78763dea 100644
--- a/tests/workspace/test_ws_household.py
+++ b/tests/workspace/test_ws_household.py
@@ -1,7 +1,10 @@
from typing import TYPE_CHECKING
import pytest
+from django import forms
from django.urls import reverse
+from strategy_field.utils import fqn
+from testutils.factories import DataCheckerFactory
from testutils.perms import user_grant_permissions
from testutils.utils import select_office
@@ -9,10 +12,11 @@
if TYPE_CHECKING:
from django_webtest.pytest_plugin import MixinWithInstanceVariables
+ from hope_flex_fields.models import DataChecker
from responses import RequestsMock
from testutils.types import CWTestApp
- from country_workspace.workspaces.models import CountryHousehold, CountryProgram
+ from country_workspace.workspaces.models import CountryHousehold, CountryIndividual, CountryProgram
pytestmark = [pytest.mark.admin, pytest.mark.smoke, pytest.mark.django_db]
@@ -45,6 +49,18 @@ def household(program):
return CountryHouseholdFactory(batch__program=program, batch__country_office=program.country_office)
+@pytest.fixture
+def individual(household: "CountryHousehold") -> "CountryIndividual":
+ from testutils.factories import CountryIndividualFactory
+
+ return CountryIndividualFactory(
+ batch=household.batch,
+ household=household,
+ batch__program=household.batch.program,
+ batch__country_office=household.batch.program.country_office,
+ )
+
+
@pytest.fixture
def app(django_app_factory: "MixinWithInstanceVariables", mocked_responses: "RequestsMock") -> "CWTestApp":
from testutils.factories import SuperUserFactory
@@ -88,19 +104,48 @@ def test_hh_validate_single(app: "CWTestApp", household: "CountryHousehold") ->
assert res.status_code == 200
-def test_hh_validate_program(app: "CWTestApp", household: "CountryHousehold") -> None:
- with select_office(app, household.country_office, household.program):
- with user_grant_permissions(app._user, ["workspaces.change_countryhousehold"], household.program):
- url = reverse("workspace:workspaces_countryhousehold_changelist")
- res = app.get(url)
- res.click("Validate Programme").follow()
- household.refresh_from_db()
- assert household.last_checked
-
-
def test_hh_update_single(app: "CWTestApp", household: "CountryHousehold") -> None:
with select_office(app, household.country_office, household.program):
with user_grant_permissions(app._user, ["workspaces.change_countryhousehold"], household.program):
url = reverse("workspace:workspaces_countryhousehold_change", args=[household.pk])
res = app.get(url)
assert res.status_code == 200
+
+
+def test_hh_validate_program(app: "CWTestApp", individual: "CountryIndividual"):
+ program: "CountryProgram" = individual.program
+ assert not individual.last_checked
+
+ with select_office(app, program.country_office, program):
+ url = reverse("workspace:workspaces_countryhousehold_changelist")
+ res = app.get(url)
+ res.click("Validate Program").follow()
+
+ individual.refresh_from_db()
+ assert individual.household.last_checked
+ assert individual.last_checked
+
+
+@pytest.fixture
+def hh(household) -> "CountryHousehold":
+ dc: DataChecker = DataCheckerFactory(fields=[("address", fqn(forms.CharField))])
+ fld = dc.fieldsets.first().fields.get(name="address")
+ fld.attrs["required"] = True
+ fld.save()
+ household.flex_fields["address"] = None
+ household.save()
+
+ household.program.household_checker = dc
+ household.program.save()
+
+ return household
+
+
+def test_hh_validate(app: "CWTestApp", hh: "CountryHousehold"):
+ assert not hh.validate_with_checker()
+ assert hh.errors == {"address": ["This field is required."]}
+
+ hh.flex_fields["address"] = "abc"
+ hh.save()
+ assert hh.validate_with_checker()
+ assert hh.errors == {}
diff --git a/tests/workspace/test_ws_import.py b/tests/workspace/test_ws_import.py
index 8bde4743..ab82227e 100644
--- a/tests/workspace/test_ws_import.py
+++ b/tests/workspace/test_ws_import.py
@@ -1,15 +1,14 @@
+import re
from typing import TYPE_CHECKING, Any
-import responses
+
import pytest
-import re
+import responses
+from constance import config
from django.urls import reverse
from country_workspace.state import state
-from constance import config
-
from tests.contrib.aurora import stub
-
if TYPE_CHECKING:
from django_webtest import DjangoTestApp
from django_webtest.pytest_plugin import MixinWithInstanceVariables
diff --git a/worker b/worker
deleted file mode 100644
index 3a80d5fc..00000000
Binary files a/worker and /dev/null differ