From 2811cb1b827a46fd7c6313dff67b105d247a651c Mon Sep 17 00:00:00 2001 From: Alastair Deng Date: Thu, 29 Aug 2024 19:30:11 -0700 Subject: [PATCH 01/11] project_list includes and displays inactive projects with warning badge --- .../core/portal/templates/portal/authorized_home.html | 2 ++ coldfront/core/portal/views.py | 8 +++++--- 2 files changed, 7 insertions(+), 3 deletions(-) diff --git a/coldfront/core/portal/templates/portal/authorized_home.html b/coldfront/core/portal/templates/portal/authorized_home.html index 1c3a5ea5c..fe37885fb 100644 --- a/coldfront/core/portal/templates/portal/authorized_home.html +++ b/coldfront/core/portal/templates/portal/authorized_home.html @@ -168,6 +168,8 @@

My {{ PROGRAM_NAME_SHORT }} Cluster Projects »

Pending {% elif 'Processing' in project.display_status %} Processing + {% elif 'Inactive' in project.display_status %} + Inactive {% else %} No Access {% endif %} diff --git a/coldfront/core/portal/views.py b/coldfront/core/portal/views.py index 54591915a..61dc55ec0 100644 --- a/coldfront/core/portal/views.py +++ b/coldfront/core/portal/views.py @@ -17,7 +17,7 @@ generate_publication_by_year_chart_data, generate_resources_chart_data, generate_total_grants_by_agency_chart_data) -from coldfront.core.project.models import Project, ProjectUserJoinRequest +from coldfront.core.project.models import Project, ProjectStatusChoice, ProjectUserJoinRequest from coldfront.core.project.models import ProjectUserJoinRequest from coldfront.core.project.models import ProjectUserRemovalRequest @@ -32,7 +32,7 @@ def home(request): if request.user.is_authenticated: template_name = 'portal/authorized_home.html' project_list = Project.objects.filter( - (Q(status__name__in=['New', 'Active', ]) & + (Q(status__name__in=['New', 'Active', 'Inactive']) & Q(projectuser__user=request.user) & Q(projectuser__status__name__in=['Active', 'Pending - Remove'])) ).distinct().order_by('name') @@ -47,7 +47,9 @@ def home(request): for project in project_list: project.display_status = access_states.get(project, None) - if (project.display_status is not None and + if project.status == ProjectStatusChoice.objects.get(name='Inactive'): + project.display_status = 'Inactive' + elif (project.display_status is not None and 'Active' in project.display_status): context['cluster_username'] = request.user.username From 4107032f1de8b29f7a9286666835ee375305c5e6 Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Mon, 9 Sep 2024 15:34:15 -0700 Subject: [PATCH 02/11] Add reusable templates for cluster-access badges and (i) popups --- .../allocation/cluster_access_badge.html | 38 +++++++++++++++++++ .../templates/portal/info_hover_popup.html | 4 ++ 2 files changed, 42 insertions(+) create mode 100644 coldfront/core/allocation/templates/allocation/cluster_access_badge.html create mode 100644 coldfront/core/portal/templates/portal/info_hover_popup.html diff --git a/coldfront/core/allocation/templates/allocation/cluster_access_badge.html b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html new file mode 100644 index 000000000..525ce4db6 --- /dev/null +++ b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html @@ -0,0 +1,38 @@ +{% if status == 'Active' %} + + Active + +{% elif status == 'Pending - Add' %} + + Pending + + {% include "portal/info_hover_popup.html" with title="Pending" content="Administrators will review the user's request to access the project on the cluster, and configure their cluster account accordingly." %} +{% elif status == 'Pending - Remove' %} + + Pending Removal + + {% include "portal/info_hover_popup.html" with title="Pending Removal" content="The user is being removed from the project. Administrators will configure their cluster account accordingly." %} +{% elif status == 'Processing' %} + {% comment %} + In practice, this status is not set. + {% endcomment %} + + Processing + + {% include "portal/info_hover_popup.html" with title="Processing" content="Administrators have approved the user's request to access the project on the cluster, and are currently configuring their cluster account accordingly." %} +{% elif status == "Denied" %} + + Denied + + {% include "portal/info_hover_popup.html" with title="Denied" content="Administrators have denied the user's request to access the project on the cluster. Please contact us for further assistance." %} +{% elif status == "None" %} + + No Access + + {% include "portal/info_hover_popup.html" with title="No Access" content="The user has not requested access to the project on the cluster." %} +{% else %} + + Error + + {% include "portal/info_hover_popup.html" with title="Error" content="Error. Please contact an administrator." %} +{% endif %} diff --git a/coldfront/core/portal/templates/portal/info_hover_popup.html b/coldfront/core/portal/templates/portal/info_hover_popup.html new file mode 100644 index 000000000..46858d969 --- /dev/null +++ b/coldfront/core/portal/templates/portal/info_hover_popup.html @@ -0,0 +1,4 @@ + + {{ title }} Info Hover Popup Text + + From 0c1d9c69eab477ac86c4a3544860cfb27e714b21 Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Mon, 9 Sep 2024 15:43:57 -0700 Subject: [PATCH 03/11] Include popups explaining 'Access to Project on Cluster' statuses on home page --- .../templates/portal/authorized_home.html | 13 +------ coldfront/core/portal/views.py | 37 ++++++++++++------- 2 files changed, 25 insertions(+), 25 deletions(-) diff --git a/coldfront/core/portal/templates/portal/authorized_home.html b/coldfront/core/portal/templates/portal/authorized_home.html index 1c3a5ea5c..22cd3de1c 100644 --- a/coldfront/core/portal/templates/portal/authorized_home.html +++ b/coldfront/core/portal/templates/portal/authorized_home.html @@ -160,17 +160,7 @@

My {{ PROGRAM_NAME_SHORT }} Cluster Projects »

{{ project.cluster_name|upper }} - {% if project.name in pending_removal_request_projects %} - Pending Removal - {% elif 'Active' in project.display_status %} - Active - {% elif 'Pending' in project.display_status %} - Pending - {% elif 'Processing' in project.display_status %} - Processing - {% else %} - No Access - {% endif %} + {% include "allocation/cluster_access_badge.html" with status=project.display_status %} {% endfor %} @@ -262,6 +252,7 @@

Allocations »

diff --git a/coldfront/core/portal/views.py b/coldfront/core/portal/views.py index 54591915a..60ca4269f 100644 --- a/coldfront/core/portal/views.py +++ b/coldfront/core/portal/views.py @@ -28,6 +28,27 @@ def home(request): + def _compute_project_user_cluster_access_statuses(_user): + """Return a dict mapping each Project object that the given User + is associated with to a str describing the user's access to the + cluster under the project.""" + statuses = {} + + cluster_access_attributes = AllocationUserAttribute.objects.filter( + allocation_attribute_type__name='Cluster Account Status', + allocation_user__user=_user) + for attribute in cluster_access_attributes: + _project = attribute.allocation.project + statuses[_project] = attribute.value + + for project_user_removal_request in \ + ProjectUserRemovalRequest.objects.filter( + project_user__user=_user, status__name='Pending'): + _project = project_user_removal_request.project_user.project + statuses[_project] = 'Pending - Remove' + + return statuses + context = {} if request.user.is_authenticated: template_name = 'portal/authorized_home.html' @@ -37,13 +58,8 @@ def home(request): Q(projectuser__status__name__in=['Active', 'Pending - Remove'])) ).distinct().order_by('name') - cluster_access_attributes = AllocationUserAttribute.objects.filter(allocation_attribute_type__name='Cluster Account Status', - allocation_user__user=request.user) - access_states = {} - for attribute in cluster_access_attributes: - project = attribute.allocation.project - status = attribute.value - access_states[project] = status + access_states = _compute_project_user_cluster_access_statuses( + request.user) for project in project_list: project.display_status = access_states.get(project, None) @@ -74,13 +90,6 @@ def home(request): context['num_join_requests'] = num_join_requests - context['pending_removal_request_projects'] = \ - [removal_request.project_user.project.name - for removal_request in - ProjectUserRemovalRequest.objects.filter( - Q(project_user__user__username=request.user.username) & - Q(status__name='Pending'))] - else: template_name = 'portal/nonauthorized_home.html' From 5dee1f8825a7e708b84e1a0f645f80a20f4f6231 Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Mon, 9 Sep 2024 15:45:38 -0700 Subject: [PATCH 04/11] Include popups explaining 'Project Access on Cluster' statuses on Project detail and ProjectUser detail views --- .../templates/project/project_detail.html | 50 +++++++------------ .../project/project_user_detail.html | 31 ++++-------- coldfront/core/project/views.py | 9 ++-- 3 files changed, 34 insertions(+), 56 deletions(-) diff --git a/coldfront/core/project/templates/project/project_detail.html b/coldfront/core/project/templates/project/project_detail.html index 80f214a89..74e118cfa 100644 --- a/coldfront/core/project/templates/project/project_detail.html +++ b/coldfront/core/project/templates/project/project_detail.html @@ -1,4 +1,5 @@ {% extends "common/base.html" %} +{% load common_tags %} {% load crispy_forms_tags %} {% load humanize %} {% load static %} @@ -290,14 +291,13 @@

Email Role - - Role Help Text - - + {% include "portal/info_hover_popup.html" with title="Role" content="Manager role grants user access to add/remove users, and allocations to the project." %} Cluster Username - Project Access on Cluster - + + Project Access on Cluster + {% include "portal/info_hover_popup.html" with title="Project Access on Cluster" content="This denotes whether the user's cluster account has been configured to have access to the project." %} + {% if is_allowed_to_update_project %} Actions {% endif %} @@ -319,31 +319,19 @@

{% endif %} - {% if user.cluster_access_status == "Active" %} - - Active - - {% elif user.cluster_access_status == "Pending - Add" %} - - Pending - - {% elif user.cluster_access_status == "Processing" %} - - Processing - - {% else %} - - No Access - - {% if cluster_accounts_requestable and is_allowed_to_update_project and user.user in project.pis or user.user in project.managers %} -

-
- {% csrf_token %} - -
+ {% include "allocation/cluster_access_badge.html" with status=user.cluster_access_status %} + {% if user.cluster_access_status == "None" %} + {% if user.role.name == "Principal Investigator" or user.role.name == "Manager" %} + {% if cluster_accounts_requestable and is_allowed_to_update_project %} +

+
+ {% csrf_token %} + +
+ {% endif %} {% endif %} {% endif %} diff --git a/coldfront/core/project/templates/project/project_user_detail.html b/coldfront/core/project/templates/project/project_user_detail.html index f67a0a7a1..5d4471585 100644 --- a/coldfront/core/project/templates/project/project_user_detail.html +++ b/coldfront/core/project/templates/project/project_user_detail.html @@ -1,4 +1,5 @@ {% extends "common/base.html" %} +{% load common_tags %} {% load crispy_forms_tags %} {% load static %} @@ -102,34 +103,18 @@

Project: {{project_obj.name}}


Project Access on Cluster: - {% if cluster_access_status == 'Active' %} - - Active - - {% elif cluster_access_status == 'Pending - Add' %} - - Pending - - {% elif cluster_access_status == 'Processing' %} - - Processing - - {% elif cluster_access_status == 'None' %} - - No Access - - {% if project_user_obj.user in project_user_obj.project.pis or project_user_obj.user in project_user_obj.project.managers %} + {% include "allocation/cluster_access_badge.html" with status=cluster_access_status %} + {% if cluster_access_status == "None" %} + {% if project_user_obj.role.name == "Principal Investigator" or project_user_obj.role.name == "Manager" %}

{% csrf_token %}
{% endif %} - {% else %} - Failed to retrieve status. Please contact an administrator. {% endif %} @@ -139,4 +124,10 @@

Project: {{project_obj.name}}


+ + {% endblock%} diff --git a/coldfront/core/project/views.py b/coldfront/core/project/views.py index 17fdb7533..2bb3979fc 100644 --- a/coldfront/core/project/views.py +++ b/coldfront/core/project/views.py @@ -20,7 +20,6 @@ AllocationStatusChoice, AllocationUserAttribute, AllocationUserStatusChoice) -from coldfront.core.allocation.utils import get_allocation_user_cluster_access_status from coldfront.core.allocation.utils import get_project_compute_allocation from coldfront.core.allocation.utils import get_project_compute_resource_name # from coldfront.core.grant.models import Grant @@ -138,8 +137,7 @@ def get_context_data(self, **kwargs): allocation_obj.allocationuserattribute_set.select_related( 'allocation_user__user' ).filter( - allocation_attribute_type__name='Cluster Account Status', - value__in=['Pending - Add', 'Processing', 'Active']) + allocation_attribute_type__name='Cluster Account Status') for status in statuses: username = status.allocation_user.user.username cluster_access_statuses[username] = status.value @@ -1109,8 +1107,9 @@ def get(self, request, *args, **kwargs): else: try: cluster_access_status = \ - get_allocation_user_cluster_access_status( - allocation_obj, project_user_obj.user).value + allocation_obj.allocationuserattribute_set.get( + allocation_user__user=project_user_obj.user, + allocation_attribute_type__name='Cluster Account Status').value except AllocationUserAttribute.DoesNotExist: cluster_access_status = 'None' except AllocationUserAttribute.MultipleObjectsReturned: From 6adce86ce05df32f06e29050050928bd71f0d831 Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Mon, 9 Sep 2024 15:46:39 -0700 Subject: [PATCH 05/11] Remove now unused utility function --- coldfront/core/allocation/utils.py | 7 ------- 1 file changed, 7 deletions(-) diff --git a/coldfront/core/allocation/utils.py b/coldfront/core/allocation/utils.py index 4566ea253..a328e6eab 100644 --- a/coldfront/core/allocation/utils.py +++ b/coldfront/core/allocation/utils.py @@ -153,13 +153,6 @@ def set_allocation_user_attribute_value(allocation_user_obj, type_name, value): return allocation_user_attribute -def get_allocation_user_cluster_access_status(allocation_obj, user_obj): - return allocation_obj.allocationuserattribute_set.get( - allocation_user__user=user_obj, - allocation_attribute_type__name='Cluster Account Status', - value__in=['Pending - Add', 'Processing', 'Active']) - - def get_project_compute_resource_name(project_obj): """Return the name of the '{cluster_name} Compute' Resource that corresponds to the given Project. From f9f7c850139e9b2e617b102ca259f8282118fdc9 Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Tue, 10 Sep 2024 09:07:48 -0700 Subject: [PATCH 06/11] Maintain quote consistency in template --- .../templates/allocation/cluster_access_badge.html | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/coldfront/core/allocation/templates/allocation/cluster_access_badge.html b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html index 525ce4db6..23f2f5a29 100644 --- a/coldfront/core/allocation/templates/allocation/cluster_access_badge.html +++ b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html @@ -1,18 +1,18 @@ -{% if status == 'Active' %} +{% if status == "Active" %} Active -{% elif status == 'Pending - Add' %} +{% elif status == "Pending - Add" %} Pending {% include "portal/info_hover_popup.html" with title="Pending" content="Administrators will review the user's request to access the project on the cluster, and configure their cluster account accordingly." %} -{% elif status == 'Pending - Remove' %} +{% elif status == "Pending - Remove" %} Pending Removal {% include "portal/info_hover_popup.html" with title="Pending Removal" content="The user is being removed from the project. Administrators will configure their cluster account accordingly." %} -{% elif status == 'Processing' %} +{% elif status == "Processing" %} {% comment %} In practice, this status is not set. {% endcomment %} From 8ddd9197d0910d0ed8e39d74f3650ade1f46258d Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Tue, 10 Sep 2024 09:09:10 -0700 Subject: [PATCH 07/11] Change default cluster access status & how cluster username is determined on home page --- coldfront/core/portal/views.py | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/coldfront/core/portal/views.py b/coldfront/core/portal/views.py index 60ca4269f..397fae100 100644 --- a/coldfront/core/portal/views.py +++ b/coldfront/core/portal/views.py @@ -12,6 +12,7 @@ AllocationUser, AllocationUserAttribute) from coldfront.core.allocation.utils import get_project_compute_resource_name +from coldfront.core.allocation.utils import has_cluster_access # from coldfront.core.grant.models import Grant from coldfront.core.portal.utils import (generate_allocations_chart_data, generate_publication_by_year_chart_data, @@ -62,14 +63,13 @@ def _compute_project_user_cluster_access_statuses(_user): request.user) for project in project_list: - project.display_status = access_states.get(project, None) - if (project.display_status is not None and - 'Active' in project.display_status): - context['cluster_username'] = request.user.username - + project.display_status = access_states.get(project, 'None') resource_name = get_project_compute_resource_name(project) project.cluster_name = resource_name.replace(' Compute', '') + if has_cluster_access(request.user): + context['cluster_username'] = request.user.username + allocation_list = Allocation.objects.filter( Q(status__name__in=['Active', 'New', 'Renewal Requested', ]) & Q(project__status__name__in=['Active', 'New']) & From 7f96310d8a907d624a2ffb8f6f430ebba79c5f64 Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Tue, 10 Sep 2024 09:12:32 -0700 Subject: [PATCH 08/11] Remove unnecessary loading of common_tags --- coldfront/core/project/templates/project/project_detail.html | 1 - .../core/project/templates/project/project_user_detail.html | 1 - 2 files changed, 2 deletions(-) diff --git a/coldfront/core/project/templates/project/project_detail.html b/coldfront/core/project/templates/project/project_detail.html index 74e118cfa..dd52087b1 100644 --- a/coldfront/core/project/templates/project/project_detail.html +++ b/coldfront/core/project/templates/project/project_detail.html @@ -1,5 +1,4 @@ {% extends "common/base.html" %} -{% load common_tags %} {% load crispy_forms_tags %} {% load humanize %} {% load static %} diff --git a/coldfront/core/project/templates/project/project_user_detail.html b/coldfront/core/project/templates/project/project_user_detail.html index 5d4471585..a6f34e26f 100644 --- a/coldfront/core/project/templates/project/project_user_detail.html +++ b/coldfront/core/project/templates/project/project_user_detail.html @@ -1,5 +1,4 @@ {% extends "common/base.html" %} -{% load common_tags %} {% load crispy_forms_tags %} {% load static %} From b11dfa1f5294258c634ed414b0fc3438da9e1f3c Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Tue, 10 Sep 2024 10:06:02 -0700 Subject: [PATCH 09/11] Add comments describing how to use the templates for cluster access badges and info icons --- .../allocation/cluster_access_badge.html | 19 +++++++++++++++++++ .../templates/portal/info_hover_popup.html | 16 ++++++++++++++++ 2 files changed, 35 insertions(+) diff --git a/coldfront/core/allocation/templates/allocation/cluster_access_badge.html b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html index 23f2f5a29..3ef324dbb 100644 --- a/coldfront/core/allocation/templates/allocation/cluster_access_badge.html +++ b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html @@ -1,3 +1,22 @@ +{% comment %} + +Display a badge, info icon, and popup denoting a user's access to the cluster +under a project, based on a string `status`, which is expected to be one of the +following values: + +"Active" +"Pending - Add" +"Pending - Remove" +"Processing" +"Denied" +"None" + +If a different value is given, display an "Error" badge. + +Usage: {% include "allocation/cluster_access_badge.html" with status="status" %} + +{% endcomment %} + {% if status == "Active" %} Active diff --git a/coldfront/core/portal/templates/portal/info_hover_popup.html b/coldfront/core/portal/templates/portal/info_hover_popup.html index 46858d969..2d0b59516 100644 --- a/coldfront/core/portal/templates/portal/info_hover_popup.html +++ b/coldfront/core/portal/templates/portal/info_hover_popup.html @@ -1,3 +1,19 @@ +{% comment %} + +Display an info icon that, when hovered over, displays a popup that has a title +and content (text). + +Usage: {% include "portal/info_hover_popup.html" with title="title" content="content" %} + +The following jQuery code must be included in the calling template for the popup +to appear: + +$(document).ready(function(){ + $('[data-toggle="popover"]').popover(); +}); + +{% endcomment %} + {{ title }} Info Hover Popup Text From 29da4e591b985ecbeb707c0b2054edee00815c57 Mon Sep 17 00:00:00 2001 From: Matthew Li Date: Tue, 10 Sep 2024 10:22:22 -0700 Subject: [PATCH 10/11] Remove case for 'Processing' entirely given that the status is not expected to occur --- .../templates/allocation/cluster_access_badge.html | 9 --------- 1 file changed, 9 deletions(-) diff --git a/coldfront/core/allocation/templates/allocation/cluster_access_badge.html b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html index 3ef324dbb..99bf926b5 100644 --- a/coldfront/core/allocation/templates/allocation/cluster_access_badge.html +++ b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html @@ -7,7 +7,6 @@ "Active" "Pending - Add" "Pending - Remove" -"Processing" "Denied" "None" @@ -31,14 +30,6 @@ Pending Removal {% include "portal/info_hover_popup.html" with title="Pending Removal" content="The user is being removed from the project. Administrators will configure their cluster account accordingly." %} -{% elif status == "Processing" %} - {% comment %} - In practice, this status is not set. - {% endcomment %} - - Processing - - {% include "portal/info_hover_popup.html" with title="Processing" content="Administrators have approved the user's request to access the project on the cluster, and are currently configuring their cluster account accordingly." %} {% elif status == "Denied" %} Denied From 82792383f7e3bc3d43ac4b50deeb925ffd2ed1fc Mon Sep 17 00:00:00 2001 From: Alastair Deng Date: Tue, 17 Sep 2024 14:04:51 -0500 Subject: [PATCH 11/11] badge and popup on home page for inactive projects --- coldfront/core/portal/templates/portal/authorized_home.html | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/coldfront/core/portal/templates/portal/authorized_home.html b/coldfront/core/portal/templates/portal/authorized_home.html index 22cd3de1c..15cfd1a7b 100644 --- a/coldfront/core/portal/templates/portal/authorized_home.html +++ b/coldfront/core/portal/templates/portal/authorized_home.html @@ -150,7 +150,11 @@

My {{ PROGRAM_NAME_SHORT }} Cluster Projects »

- {{ project.name }} + {{ project.name }} + {% if project.status.name == 'Inactive' %} + {{project.status}} + {% include "portal/info_hover_popup.html" with title="Inactive" content="This project's computing allowance must be renewed before jobs may be submitted under it." %} + {% endif %} {% if project.needs_review %}