Skip to content

Commit

Permalink
Merge branch '2.x' into speed-up-analysis-verification
Browse files Browse the repository at this point in the history
  • Loading branch information
ramonski authored Nov 19, 2024
2 parents c106b06 + 7be1e36 commit 1e66ecf
Show file tree
Hide file tree
Showing 17 changed files with 662 additions and 68 deletions.
2 changes: 2 additions & 0 deletions CHANGES.rst
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@ Changelog
------------------

- #2643 Improve performance of analysis verification
- #2624 Added "Maximum holding time" setting to services and analyses
- #2637 Do not remove inactive services from profiles and templates
- #2642 Fix Attribute Error in Upgrade Step 2619
- #2641 Fix AttributeError on rejection of samples without a contact set
- #2640 Fix missing custom transitions via adapter in Worksheet's analyses
Expand Down
69 changes: 69 additions & 0 deletions src/bika/lims/browser/analyses/view.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
from copy import copy
from copy import deepcopy
from datetime import datetime
from datetime import timedelta
from operator import itemgetter

from bika.lims import api
Expand All @@ -38,9 +39,11 @@
from bika.lims.interfaces import IFieldIcons
from bika.lims.interfaces import IReferenceAnalysis
from bika.lims.interfaces import IRoutineAnalysis
from bika.lims.interfaces import ISubmitted
from bika.lims.utils import check_permission
from bika.lims.utils import format_supsub
from bika.lims.utils import formatDecimalMark
from bika.lims.utils import get_fas_ico
from bika.lims.utils import get_image
from bika.lims.utils import get_link
from bika.lims.utils import get_link_for
Expand Down Expand Up @@ -790,6 +793,8 @@ def folderitem(self, obj, item, index):
self._folder_item_remarks(obj, item)
# Renders the analysis conditions
self._folder_item_conditions(obj, item)
# Fill maximum holding time warnings
self._folder_item_holding_time(obj, item)

return item

Expand Down Expand Up @@ -1698,6 +1703,70 @@ def to_str(condition):
service = item["replace"].get("Service") or item["Service"]
item["replace"]["Service"] = "<br/>".join([service, conditions])

def _folder_item_holding_time(self, analysis_brain, item):
"""Adds an icon to the item dictionary if no result has been submitted
for the analysis and the holding time has passed or is about to expire.
It also displays the icon if the result was recorded after the holding
time limit.
"""
analysis = self.get_object(analysis_brain)
if not IRoutineAnalysis.providedBy(analysis):
return

# get the maximum holding time for this analysis
max_holding_time = analysis.getMaxHoldingTime()
if not max_holding_time:
return

# get the datetime from which the max holding time is computed
start_date = analysis.getDateSampled()
start_date = dtime.to_dt(start_date)
if not start_date:
return

# get the timezone of the start date for correct comparisons
timezone = dtime.get_timezone(start_date)

# calculate the maximum holding date
delta = timedelta(minutes=api.to_minutes(**max_holding_time))
max_holding_date = dtime.to_ansi(start_date + delta)

# maybe the result was captured past the holding time
if ISubmitted.providedBy(analysis):
captured = analysis.getResultCaptureDate()
captured = dtime.to_ansi(captured, timezone=timezone)
if captured > max_holding_date:
msg = _("The result was captured past the holding time limit.")
icon = get_fas_ico("exclamation-triangle",
css_class="text-danger",
title=t(msg))
self._append_html_element(item, "ResultCaptureDate", icon)
return

# not yet submitted, maybe the holding time expired
now = dtime.to_ansi(dtime.now(), timezone=timezone)
if now > max_holding_date:
msg = _("The holding time for this sample and analysis has "
"expired. Proceeding with the analysis may compromise the "
"reliability of the results.")
icon = get_fas_ico("exclamation-triangle",
css_class="text-danger",
title=t(msg))
self._append_html_element(item, "ResultCaptureDate", icon)
return

# or maybe is about to expire
soon = dtime.to_ansi(dtime.now(), timezone=timezone)
if soon > max_holding_date:
msg = _("The holding time for this sample and analysis is about "
"to expire. Please complete the analysis as soon as "
"possible to ensure data accuracy and reliability.")
icon = get_fas_ico("exclamation-triangle",
css_class="text-warning",
title=t(msg))
self._append_html_element(item, "ResultCaptureDate", icon)
return

def is_method_required(self, analysis):
"""Returns whether the render of the selection list with methods is
required for the method passed-in, even if only option "None" is
Expand Down
68 changes: 68 additions & 0 deletions src/bika/lims/browser/analysisrequest/add2.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
import json
from collections import OrderedDict
from datetime import datetime
from datetime import timedelta

import six
import transaction
Expand Down Expand Up @@ -50,7 +51,9 @@
from Products.CMFPlone.utils import safe_unicode
from Products.Five.browser import BrowserView
from Products.Five.browser.pagetemplatefile import ViewPageTemplateFile
from senaite.core.api import dtime
from senaite.core.catalog import CONTACT_CATALOG
from senaite.core.catalog import SETUP_CATALOG
from senaite.core.p3compat import cmp
from senaite.core.permissions import TransitionMultiResults
from zope.annotation.interfaces import IAnnotations
Expand Down Expand Up @@ -908,6 +911,7 @@ def get_service_info(self, obj):
"category": obj.getCategoryTitle(),
"poc": obj.getPointOfCapture(),
"conditions": self.get_conditions_info(obj),
"max_holding_time": obj.getMaxHoldingTime(),
})

dependencies = get_calculation_dependencies_for(obj).values()
Expand Down Expand Up @@ -1217,11 +1221,75 @@ def ajax_recalculate_records(self):
dependencies = self.get_unmet_dependencies_info(metadata)
metadata.update(dependencies)

# services conducted beyond the holding time limit
beyond = self.get_services_beyond_holding_time(record)
metadata["beyond_holding_time"] = beyond

# Set the metadata for current sample number (column)
out[num_sample] = metadata

return out

@viewcache.memoize
def get_services_max_holding_time(self):
"""Returns a dict where the key is the uid of active services and the
value is a dict representing the maximum holding time in days, hours
and minutes. The dictionary only contains uids for services that have
a valid maximum holding time set
"""
services = {}
query = {
"portal_type": "AnalysisService",
"point_of_capture": "lab",
"is_active": True,
}
brains = api.search(query, SETUP_CATALOG)
for brain in brains:
obj = api.get_object(brain)
max_holding_time = obj.getMaxHoldingTime()
if max_holding_time:
uid = api.get_uid(brain)
services[uid] = max_holding_time.copy()

return services

def get_services_beyond_holding_time(self, record):
"""Return a list with the uids of the services that cannot be selected
because would be conducted past the holding time limit
"""
# get the date to start count from
start_date = self.get_start_holding_date(record)
if not start_date:
return []

# get the timezone of the start date for correct comparisons
tz = dtime.get_timezone(start_date)

uids = []

# get the max holding times grouped by service uid
services = self.get_services_max_holding_time()
for uid, max_holding_time in services.items():

# calculate the maximum holding date
delta = timedelta(minutes=api.to_minutes(**max_holding_time))
max_holding_date = start_date + delta

# TypeError: can't compare offset-naive and offset-aware datetimes
max_date = dtime.to_ansi(max_holding_date)
now = dtime.to_ansi(dtime.now(), timezone=tz)
if now > max_date:
uids.append(uid)

return uids

def get_start_holding_date(self, record):
"""Returns the datetime used to calculate the holding time limit,
typically the sample collection date.
"""
sampled = record.get("DateSampled")
return dtime.to_dt(sampled)

def get_record_metadata(self, record):
"""Returns the metadata for the record passed in
"""
Expand Down
10 changes: 10 additions & 0 deletions src/bika/lims/browser/analysisrequest/templates/ar_add2.pt
Original file line number Diff line number Diff line change
Expand Up @@ -542,6 +542,16 @@
&#128274;
</div>

<!-- Beyond holding time box -->
<div tal:attributes="uid python:service_uid;
arnum python:arnum;
id python:'{}-{}-beyondholdingtime'.format(service_uid, arnum);
class python:'service-beyondholdingtime {}-beyondholdingtime'.format(service_uid);"
title="This service cannot be selected to prevent the analysis from being conducted beyond the analytical holding time."
i18n:attributes="title">
&#128683;
</div>

<!-- Service checkbox -->
<div tal:attributes="id python:'{}-{}-analysisservice'.format(service_uid, arnum);
class python:'analysisservice {}-analysisservice'.format(service_uid);">
Expand Down
40 changes: 40 additions & 0 deletions src/bika/lims/content/abstractbaseanalysis.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
# Some rights reserved, see README and LICENSE.

from AccessControl import ClassSecurityInfo
from bika.lims import api
from bika.lims import bikaMessageFactory as _
from bika.lims.browser.fields import DurationField
from bika.lims.browser.fields import UIDReferenceField
Expand Down Expand Up @@ -365,6 +366,30 @@
)
)

MaxHoldingTime = DurationField(
'MaxHoldingTime',
schemata="Analysis",
widget=DurationWidget(
label=_(
u"label_analysis_maxholdingtime",
default=u"Maximum holding time"
),
description=_(
u"description_analysis_maxholdingtime",
default=u"This service will not appear for selection on the "
u"sample registration form if the elapsed time since "
u"sample collection exceeds the holding time limit. "
u"Exceeding this time limit may result in unreliable or "
u"compromised data, as the integrity of the sample can "
u"degrade over time. Consequently, any results obtained "
u"after this period may not accurately reflect the "
u"sample's true composition, impacting data validity. "
u"Note: This setting does not affect the test's "
u"availability in the 'Manage Analyses' view."
)
)
)

# The amount of difference allowed between this analysis, and any duplicates.
DuplicateVariation = FixedPointField(
'DuplicateVariation',
Expand Down Expand Up @@ -789,6 +814,7 @@
Instrument,
Method,
MaxTimeAllowed,
MaxHoldingTime,
DuplicateVariation,
Accredited,
PointOfCapture,
Expand Down Expand Up @@ -1079,6 +1105,20 @@ def getMaxTimeAllowed(self):
tat = self.Schema().getField("MaxTimeAllowed").get(self)
return tat or self.bika_setup.getDefaultTurnaroundTime()

@security.public
def getMaxHoldingTime(self):
"""Returns the maximum time since it the sample was collected for this
test/service to become available on sample creation. Returns None if no
positive maximum hold time is set. Otherwise, returns a dict with the
keys "days", "hours" and "minutes"
"""
max_hold_time = self.Schema().getField("MaxHoldingTime").get(self)
if not max_hold_time:
return {}
if api.to_minutes(**max_hold_time) <= 0:
return {}
return max_hold_time

# TODO Remove. ResultOptionsType field was replaced by ResulType field
def getResultOptionsType(self):
if self.getStringResult():
Expand Down
19 changes: 0 additions & 19 deletions src/bika/lims/content/analysisservice.py
Original file line number Diff line number Diff line change
Expand Up @@ -558,25 +558,6 @@ def _default_calculation_vocabulary(self):
dlist.add("", _("None"))
return dlist

def after_deactivate_transition_event(self):
"""Method triggered after a 'deactivate' transition
Removes this service from all assigned Profiles and Templates.
"""
catalog = api.get_tool(SETUP_CATALOG)

# Remove the service from profiles to which is assigned
profiles = catalog(portal_type="AnalysisProfile")
for profile in profiles:
profile = api.get_object(profile)
profile.remove_service(self)

# Remove the service from templates to which is assigned
templates = catalog(portal_type="SampleTemplate")
for template in templates:
template = api.get_object(template)
template.remove_service(self)

# XXX DECIDE IF NEEDED
# --------------------

Expand Down
10 changes: 10 additions & 0 deletions src/bika/lims/utils/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -915,3 +915,13 @@ def get_client(obj):
return obj.getClient()

return None


def get_fas_ico(icon_id, **attrs):
"""Returns a well-formed fontawesome (fas) icon
"""
css_class = attrs.pop("css_class", "")
if not icon_id.startswith("fa-"):
icon_id = "fa-%s" % icon_id
attrs["css_class"] = " ".join([css_class, "fas", icon_id]).strip()
return "<i %s/>" % render_html_attributes(**attrs)
9 changes: 0 additions & 9 deletions src/bika/lims/utils/analysisrequest.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,6 @@
from bika.lims.utils import tmpID
from bika.lims.workflow import doActionFor
from DateTime import DateTime
from Products.Archetypes.config import UID_CATALOG
from Products.Archetypes.event import ObjectInitializedEvent
from Products.CMFPlone.utils import _createObjectByType
from senaite.core.catalog import SETUP_CATALOG
Expand Down Expand Up @@ -267,14 +266,6 @@ def to_list(value):
# Convert them to a list of service uids
uids = filter(None, map(to_service_uid, uids))

# Extend with service uids from profiles
profiles = to_list(values.get("Profiles"))
if profiles:
uid_catalog = api.get_tool(UID_CATALOG)
for brain in uid_catalog(UID=profiles):
profile = api.get_object(brain)
uids.extend(profile.getServiceUIDs() or [])

# Get the service uids without duplicates, but preserving the order
return list(OrderedDict.fromkeys(uids).keys())

Expand Down
Loading

0 comments on commit 1e66ecf

Please sign in to comment.