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..99bf926b5 --- /dev/null +++ b/coldfront/core/allocation/templates/allocation/cluster_access_badge.html @@ -0,0 +1,48 @@ +{% 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" +"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 + +{% 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 == "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/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. diff --git a/coldfront/core/portal/templates/portal/authorized_home.html b/coldfront/core/portal/templates/portal/authorized_home.html index 1c3a5ea5c..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 %} @@ -160,17 +164,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 +256,7 @@

Allocations »

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..2d0b59516 --- /dev/null +++ b/coldfront/core/portal/templates/portal/info_hover_popup.html @@ -0,0 +1,20 @@ +{% 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 + + diff --git a/coldfront/core/portal/views.py b/coldfront/core/portal/views.py index 54591915a..8479bc53c 100644 --- a/coldfront/core/portal/views.py +++ b/coldfront/core/portal/views.py @@ -12,12 +12,13 @@ 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, 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 @@ -28,32 +29,47 @@ 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' 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') - 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) - 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']) & @@ -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' diff --git a/coldfront/core/project/templates/project/project_detail.html b/coldfront/core/project/templates/project/project_detail.html index 80f214a89..dd52087b1 100644 --- a/coldfront/core/project/templates/project/project_detail.html +++ b/coldfront/core/project/templates/project/project_detail.html @@ -290,14 +290,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 +318,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..a6f34e26f 100644 --- a/coldfront/core/project/templates/project/project_user_detail.html +++ b/coldfront/core/project/templates/project/project_user_detail.html @@ -102,34 +102,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 +123,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: