Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Enterprise CommCare Client Version Compliance Report #35468

Open
wants to merge 40 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 24 commits
Commits
Show all changes
40 commits
Select commit Hold shift + click to select a range
52520df
Refactor: rename function
jingcheng16 Nov 26, 2024
52b62e7
Util to tell if a version is out of date
jingcheng16 Dec 3, 2024
4c2c923
Add Commcare Version Compliance Tile
jingcheng16 Dec 3, 2024
2ee9d14
Add CommCare Version Compliance API
jingcheng16 Dec 3, 2024
5609175
Refactor: move get_report_task to the base class
jingcheng16 Dec 3, 2024
94f54c6
Show -- for the total if the account has no mobile worker yet
jingcheng16 Dec 3, 2024
9cc4bda
Move is_out_of_date above _parse_version
jingcheng16 Dec 4, 2024
7338fcc
change text and fix format
jingcheng16 Dec 4, 2024
36138f7
Remove unnecessary code
jingcheng16 Dec 4, 2024
0512997
Change private function to public since it's imported by other file
jingcheng16 Dec 4, 2024
9dc9a81
delay the translation until rendering
jingcheng16 Dec 4, 2024
344a76c
Rename metric to description
jingcheng16 Dec 4, 2024
1aec926
Percentage in enterprise console will have one digit
jingcheng16 Dec 4, 2024
180eb0c
Use elastic search
jingcheng16 Dec 6, 2024
b8cb113
Having a util function to handle the commcare version the mobile work…
jingcheng16 Dec 6, 2024
b494264
isort
jingcheng16 Dec 6, 2024
f38e1c3
Remove formatting of version in enterprise console so it's comparable
jingcheng16 Dec 7, 2024
fcf3c61
ES returning dates in iso format so add conversion to avoid type error
jingcheng16 Dec 7, 2024
d72d426
Merge branch 'master' into jc/commcare_client_version_compliance_report
mjriley Dec 9, 2024
b1a6545
flake8
jingcheng16 Dec 9, 2024
cd61d41
Change title
jingcheng16 Dec 9, 2024
57c489a
Merge branch 'master' into jc/commcare_client_version_compliance_report
jingcheng16 Dec 10, 2024
c03d909
Rename description to total_description
jingcheng16 Dec 10, 2024
5470ea5
Update label....
jingcheng16 Dec 10, 2024
4bc93c8
adjust tile layout
biyeun Dec 11, 2024
1c1abb6
Merge branch 'master' into jc/commcare_client_version_compliance_report
jingcheng16 Dec 11, 2024
0d85d21
Rename ignored variable
jingcheng16 Dec 12, 2024
34e358b
Refactor: format version only once
jingcheng16 Dec 12, 2024
88252a0
Memoize get_latest_version_at_time and use minute precision for date_…
jingcheng16 Dec 12, 2024
8b75af1
Fix UnboundLocalError
jingcheng16 Dec 12, 2024
c0fd790
Remove microseconds too
jingcheng16 Dec 12, 2024
05f6058
Fetch CommCareBuildConfig only once
jingcheng16 Dec 12, 2024
adb7d81
memoize build time to avoid repetitively fetching build for each version
jingcheng16 Dec 12, 2024
4a81844
update naming convention for elements refrenced in javascript
biyeun Dec 12, 2024
b4d59b4
This tile don't need domain_obj
jingcheng16 Dec 12, 2024
d3da1d7
Move total above rows_for_domain
jingcheng16 Dec 12, 2024
1cc3178
Use gevent pooling for parallel domain processing
jingcheng16 Dec 12, 2024
7599675
Refer domain name as domain and fix a format in template
jingcheng16 Dec 13, 2024
44d9db5
quickcache on globally scoped function
jingcheng16 Dec 13, 2024
bb87820
keep exception handling as close as possible to the operation that ca…
jingcheng16 Dec 13, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
28 changes: 28 additions & 0 deletions corehq/apps/builds/tests/test_utils.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
from django.test import SimpleTestCase
from corehq.apps.builds.utils import is_out_of_date


class TestVersionUtils(SimpleTestCase):

def test_is_out_of_date(self):
test_cases = [
# (version_in_use, latest_version, expected_result)
('2.53.0', '2.53.1', True), # Normal case - out of date
('2.53.1', '2.53.1', False), # Same version - not out of date
('2.53.2', '2.53.1', False), # Higher version - not out of date
(None, '2.53.1', False), # None version_in_use
('2.53.1', None, False), # None latest_version
('invalid', '2.53.1', False), # Invalid version string
('2.53.1', 'invalid', False), # Invalid latest version
('6', '7', True), # Normal case - app version is integer
(None, None, False), # None version_in_use and latest_version
]

for version_in_use, latest_version, expected in test_cases:
with self.subTest(version_in_use=version_in_use, latest_version=latest_version):
result = is_out_of_date(version_in_use, latest_version)
self.assertEqual(
result,
expected,
f"Expected is_out_of_date('{version_in_use}', '{latest_version}') to be {expected}"
)
50 changes: 50 additions & 0 deletions corehq/apps/builds/utils.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,7 @@
import re
from datetime import datetime

from dimagi.utils.parsing import ISO_DATETIME_FORMAT

from .models import CommCareBuild, CommCareBuildConfig

Expand Down Expand Up @@ -40,3 +43,50 @@ def extract_build_info_from_filename(content_disposition):
else:
raise ValueError('Could not find filename like {!r} in {!r}'.format(
pattern, content_disposition))


def get_latest_version_at_time(target_time):
"""
Get the latest CommCare version that was available at a given time.
Excludes superuser-only versions.
Menu items are already in chronological order (newest last).
If no target time is provided, return the latest version available now.
"""
config = CommCareBuildConfig.fetch()

if not target_time:
return config.get_default().version

if isinstance(target_time, str):
target_time = datetime.strptime(target_time, ISO_DATETIME_FORMAT)

# Iterate through menu items in reverse (newest to oldest)
for item in reversed(config.menu):
if item.superuser_only:
continue
try:
build = CommCareBuild.get_build(item.build.version, latest=True)
if build and build.time and build.time <= target_time:
return item.build.version
except KeyError:
continue

return None


def is_out_of_date(version_in_use, latest_version):
version_in_use_tuple = _parse_version(version_in_use)
latest_version_tuple = _parse_version(latest_version)
if not version_in_use_tuple or not latest_version_tuple:
return False
return version_in_use_tuple < latest_version_tuple


def _parse_version(version_str):
"""Convert version string to comparable tuple"""
if version_str:
try:
return tuple(int(n) for n in version_str.split('.'))
except (ValueError, AttributeError):
return None
return None
mjriley marked this conversation as resolved.
Show resolved Hide resolved
2 changes: 2 additions & 0 deletions corehq/apps/enterprise/api/api.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
from tastypie.api import Api

from corehq.apps.enterprise.api.resources import (
CommCareVersionComplianceResource,
DomainResource,
FormSubmissionResource,
MobileUserResource,
Expand All @@ -15,4 +16,5 @@
v1_api.register(MobileUserResource())
v1_api.register(FormSubmissionResource())
v1_api.register(ODataFeedResource())
v1_api.register(CommCareVersionComplianceResource())
v1_api.register(SMSResource())
58 changes: 25 additions & 33 deletions corehq/apps/enterprise/api/resources.py
Original file line number Diff line number Diff line change
Expand Up @@ -194,7 +194,12 @@ def get_object_list(self, request):
response=http.HttpTooManyRequests(headers={'Retry-After': self.RETRY_IN_PROGRESS_DELAY}))

def get_report_task(self, request):
raise NotImplementedError()
account = BillingAccount.get_account_by_domain(request.domain)
return generate_enterprise_report.s(
self.REPORT_SLUG,
account.id,
request.couch_user.username,
)

def _add_query_id_to_request(self, request, query_id):
if 'report' not in request.GET:
Expand All @@ -214,14 +219,6 @@ class DomainResource(ODataEnterpriseReportResource):

REPORT_SLUG = EnterpriseReport.DOMAINS

def get_report_task(self, request):
account = BillingAccount.get_account_by_domain(request.domain)
return generate_enterprise_report.s(
self.REPORT_SLUG,
account.id,
request.couch_user.username,
)

def dehydrate(self, bundle):
bundle.data['domain'] = bundle.obj[6]
bundle.data['created_on'] = self.convert_datetime(bundle.obj[0])
Expand All @@ -248,14 +245,6 @@ class WebUserResource(ODataEnterpriseReportResource):

REPORT_SLUG = EnterpriseReport.WEB_USERS

def get_report_task(self, request):
account = BillingAccount.get_account_by_domain(request.domain)
return generate_enterprise_report.s(
self.REPORT_SLUG,
account.id,
request.couch_user.username,
)

def dehydrate(self, bundle):
bundle.data['email'] = bundle.obj[0]
bundle.data['name'] = self.convert_not_available(bundle.obj[1])
Expand Down Expand Up @@ -289,14 +278,6 @@ class MobileUserResource(ODataEnterpriseReportResource):

REPORT_SLUG = EnterpriseReport.MOBILE_USERS

def get_report_task(self, request):
account = BillingAccount.get_account_by_domain(request.domain)
return generate_enterprise_report.s(
self.REPORT_SLUG,
account.id,
request.couch_user.username,
)

def dehydrate(self, bundle):
bundle.data['username'] = bundle.obj[0]
bundle.data['name'] = bundle.obj[1]
Expand Down Expand Up @@ -373,14 +354,6 @@ class ODataFeedResource(ODataEnterpriseReportResource):

REPORT_SLUG = EnterpriseReport.ODATA_FEEDS

def get_report_task(self, request):
account = BillingAccount.get_account_by_domain(request.domain)
return generate_enterprise_report.s(
self.REPORT_SLUG,
account.id,
request.couch_user.username,
)

def dehydrate(self, bundle):
bundle.data['num_feeds_used'] = bundle.obj[0]
bundle.data['num_feeds_available'] = bundle.obj[1]
Expand Down Expand Up @@ -433,3 +406,22 @@ def dehydrate(self, bundle):

def get_primary_keys(self):
return ('form_id', 'submitted',)


class CommCareVersionComplianceResource(ODataEnterpriseReportResource):
mobile_worker = fields.CharField()
project_space = fields.CharField()
latest_version_available_at_submission = fields.CharField()
version_in_use = fields.CharField()

REPORT_SLUG = EnterpriseReport.COMMCARE_VERSION_COMPLIANCE

def dehydrate(self, bundle):
bundle.data['mobile_worker'] = bundle.obj[0]
bundle.data['project_space'] = bundle.obj[1]
bundle.data['latest_version_available_at_submission'] = bundle.obj[2]
bundle.data['version_in_use'] = bundle.obj[3]
return bundle

def get_primary_keys(self):
return ('mobile_worker', 'project_space',)
80 changes: 77 additions & 3 deletions corehq/apps/enterprise/enterprise.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,25 +3,32 @@
from django.db.models import Count
from datetime import datetime, timedelta

from django.conf import settings
from django.utils.translation import gettext as _
from django.utils.translation import gettext_lazy
from django.conf import settings

from memoized import memoized

from couchforms.analytics import get_last_form_submission_received
from dimagi.utils.dates import DateSpan

from corehq.apps.enterprise.exceptions import EnterpriseReportError, TooMuchRequestedDataError
from corehq.apps.enterprise.iterators import raise_after_max_elements
from corehq.apps.accounting.models import BillingAccount
from corehq.apps.accounting.utils import get_default_domain_url
from corehq.apps.app_manager.dbaccessors import get_brief_apps_in_domain
from corehq.apps.builds.utils import get_latest_version_at_time, is_out_of_date
from corehq.apps.domain.calculations import sms_in_last
from corehq.apps.domain.models import Domain
from corehq.apps.enterprise.exceptions import (
EnterpriseReportError,
TooMuchRequestedDataError,
)
from corehq.apps.enterprise.iterators import raise_after_max_elements
from corehq.apps.es import forms as form_es
from corehq.apps.es.users import UserES
from corehq.apps.export.dbaccessors import ODataExportFetcher
from corehq.apps.reports.util import (
get_commcare_version_and_date_from_last_usage,
)
from corehq.apps.sms.models import SMS, OUTGOING, INCOMING
from corehq.apps.users.dbaccessors import (
get_all_user_rows,
Expand All @@ -37,6 +44,7 @@ class EnterpriseReport(ABC):
MOBILE_USERS = 'mobile_users'
FORM_SUBMISSIONS = 'form_submissions'
ODATA_FEEDS = 'odata_feeds'
COMMCARE_VERSION_COMPLIANCE = 'commcare_version_compliance'
SMS = 'sms'

DATE_ROW_FORMAT = '%Y/%m/%d %H:%M:%S'
Expand Down Expand Up @@ -81,6 +89,8 @@ def create(cls, slug, account_id, couch_user, **kwargs):
report = EnterpriseFormReport(account, couch_user, **kwargs)
elif slug == cls.ODATA_FEEDS:
report = EnterpriseODataReport(account, couch_user, **kwargs)
elif slug == cls.COMMCARE_VERSION_COMPLIANCE:
report = EnterpriseCommCareVersionReport(account, couch_user, **kwargs)
elif slug == cls.SMS:
report = EnterpriseSMSReport(account, couch_user, **kwargs)

Expand Down Expand Up @@ -405,6 +415,70 @@ def _get_individual_export_rows(self, exports, export_line_counts):
return rows


class EnterpriseCommCareVersionReport(EnterpriseReport):
title = gettext_lazy('CommCare Version Compliance')
total_description = gettext_lazy('% of Mobile Workers on the Latest CommCare Version')

@property
def headers(self):
return [
_('Mobile Worker'),
_('Project Space'),
_('Latest Version Available at Submission'),
_('Version in Use'),
]

def rows_for_domain(self, domain_obj):
rows = []

user_query = (UserES()
.domain(domain_obj.name)
.mobile_users()
.source([
'username',
'reporting_metadata.last_submission_for_user.commcare_version',
'reporting_metadata.last_submission_for_user.submission_date',
'last_device.commcare_version',
'last_device.last_used'
]))

for user in user_query.run().hits:
last_submission = user.get('reporting_metadata', {}).get('last_submission_for_user', {})
last_device = user.get('last_device', {})

version_in_use, date_of_use = get_commcare_version_and_date_from_last_usage(last_submission,
last_device)

latest_version_at_time_of_use = get_latest_version_at_time(date_of_use)

if is_out_of_date(version_in_use, latest_version_at_time_of_use):
rows.append([
user['username'],
domain_obj.name,
latest_version_at_time_of_use,
version_in_use,
])

return rows

@property
def total(self):
total_mobile_workers = 0
total_up_to_date = 0
for domain_obj in self.domains():
domain_mobile_workers = get_mobile_user_count(domain_obj.name, include_inactive=False)
if domain_mobile_workers:
total_mobile_workers += domain_mobile_workers
total_up_to_date += domain_mobile_workers - len(self.rows_for_domain(domain_obj))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is definitely a performance concern here, as if this summary query takes 2 minutes, that is 2 minutes blocking a web worker. It's worth trying to see if Elasticsearch is any faster here. Otherwise, we may want to cache this statistic.

Copy link
Contributor Author

@jingcheng16 jingcheng16 Dec 6, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Tried on production again, it takes between 2 minutes and 3 minutes. Last time I was about to leave my apartment so I didn't wait until it finished, so I guess last time when I query couch directly maybe it longer than 3 minutes. There is improvement by using elastic search. Is the improvement acceptable? I think maybe yes? Because not every enterprise account would have that amount of mobile workers...

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If we're talking minute I think we need to think about caching or turning this into a celery task. I can imagine that multiple requests to this page could cause a disruption in our responsiveness that would be hard to diagnose

return _format_percentage_for_enterprise_tile(total_up_to_date, total_mobile_workers)


def _format_percentage_for_enterprise_tile(dividend, divisor):
if not divisor:
return '--'
return f"{dividend / divisor * 100:.1f}%"


class EnterpriseSMSReport(EnterpriseReport):
title = gettext_lazy('SMS Usage')
total_description = gettext_lazy('# of SMS Sent')
Expand Down
1 change: 1 addition & 0 deletions corehq/apps/enterprise/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -91,6 +91,7 @@ def platform_overview(request, domain):
EnterpriseReport.MOBILE_USERS,
EnterpriseReport.FORM_SUBMISSIONS,
EnterpriseReport.ODATA_FEEDS,
EnterpriseReport.COMMCARE_VERSION_COMPLIANCE,
EnterpriseReport.SMS,
)],
'metric_type': 'Platform Overview',
Expand Down
20 changes: 11 additions & 9 deletions corehq/apps/reports/standard/deployments.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,8 +12,6 @@
from couchdbkit import ResourceNotFound
from memoized import memoized

from corehq.apps.reports.filters.dates import SingleDateFilter
from corehq.util.dates import iso_string_to_date
from couchexport.export import SCALAR_NEVER_WAS
from dimagi.utils.dates import safe_strftime
from dimagi.utils.parsing import string_to_utc_datetime
Expand All @@ -35,6 +33,7 @@
from corehq.apps.locations.permissions import location_safe
from corehq.apps.reports.datatables import DataTablesColumn, DataTablesHeader
from corehq.apps.reports.exceptions import BadRequestError
from corehq.apps.reports.filters.dates import SingleDateFilter
from corehq.apps.reports.filters.select import SelectApplicationFilter
from corehq.apps.reports.filters.users import ExpandedMobileWorkerFilter
from corehq.apps.reports.generic import (
Expand All @@ -46,10 +45,14 @@
ProjectReport,
ProjectReportParametersMixin,
)
from corehq.apps.reports.util import format_datatables_data
from corehq.apps.reports.util import (
format_datatables_data,
get_commcare_version_and_date_from_last_usage,
)
from corehq.apps.users.models import CouchUser
from corehq.apps.users.util import user_display_string
from corehq.const import USER_DATE_FORMAT
from corehq.util.dates import iso_string_to_date
from corehq.util.quickcache import quickcache


Expand Down Expand Up @@ -352,11 +355,10 @@ def process_rows(self, users, fmt_for_export=False):
if last_build.get('app_id') and device and device.get('app_meta'):
device_app_meta = self.get_data_for_app(device.get('app_meta'), last_build.get('app_id'))

if last_sub and last_sub.get('commcare_version'):
commcare_version = _get_commcare_version(last_sub.get('commcare_version'))
else:
if device and device.get('commcare_version', None):
commcare_version = _get_commcare_version(device['commcare_version'])
commcare_version, last_seen = get_commcare_version_and_date_from_last_usage(last_sub,
mjriley marked this conversation as resolved.
Show resolved Hide resolved
device,
formatted=True)

if last_sub and last_sub.get('submission_date'):
last_seen = string_to_utc_datetime(last_sub['submission_date'])
if last_sync and last_sync.get('sync_date'):
Expand Down Expand Up @@ -547,7 +549,7 @@ def _get_formatted_assigned_location_names(self, user, user_loc_dict):
return format_html(f'<div>{"".join(html_nodes)}</div>')


def _get_commcare_version(app_version_info):
def format_commcare_version(app_version_info):
commcare_version = (
'CommCare {}'.format(app_version_info)
if app_version_info
Expand Down
Loading
Loading