Skip to content

Commit

Permalink
Merge pull request #20 from senaite/abbott-afinion2
Browse files Browse the repository at this point in the history
Add Abbott Afinion™ 2 Analyzer import schema
  • Loading branch information
ramonski authored Dec 13, 2024
2 parents 6b9dba9 + 3971ae2 commit 9fa2d6f
Show file tree
Hide file tree
Showing 4 changed files with 336 additions and 0 deletions.
1 change: 1 addition & 0 deletions docs/changelog.rst
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ Changelog
1.0.0 (unreleased)
------------------

- #20 Add Abbott Afinion™ 2 Analyzer import schema
- #19 Add Spotchem™EL SE-1520 import schema
- #18 Add Siemens' DCA Vantage® Analyzer import schema
- #17 Add Sysmex XP-100 import schema
Expand Down
199 changes: 199 additions & 0 deletions src/senaite/astm/instruments/abbott_afinion2.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,199 @@
# -*- coding: utf-8 -*-

from senaite.astm import records
from senaite.astm.fields import ComponentField
from senaite.astm.fields import ConstantField
from senaite.astm.fields import DateTimeField
from senaite.astm.fields import IntegerField
from senaite.astm.fields import NotUsedField
from senaite.astm.fields import SetField
from senaite.astm.fields import TextField
from senaite.astm.mapping import Component

VERSION = "1.0.0"
HEADER_RX = r".*Afinion 2 Analyzer\^"

PROCESSING_IDS = (
"P", # P: Patient measurement results
"Q", # Q: Quality control results
)

SPECIMEN_SOURCES = (
"O", # O: Other
"C", # C: Blood capillary
"V", # V: Blood venous
)

ABNORMAL_FLAGS = (
"<", # <: Less than measurement lower limit
">", # >: Higher than measurement upper limit
"L", # L: Less than normal range
"H", # H: Higher than normal range
"LL", # LL: Less than extreme range
"HH", # HH: Higher than extreme range
"!", # !: Result ambiguous
)


def get_metadata(wrapper):
"""Additional metadata
:param wrapper: The wrapper instance
:returns: dictionary of additional metadata
"""
return {
"version": VERSION,
"header_rx": HEADER_RX,
}


def get_mapping():
"""Returns the wrappers for this instrument
"""
return {
"H": HeaderRecord,
"P": PatientRecord,
"O": OrderRecord,
"R": ResultRecord,
"Q": RequestInformationRecord,
"M": ManufacturerInfoRecord,
"L": TerminatorRecord,
}


class HeaderRecord(records.HeaderRecord):
"""Message Header Record (H)
This record must always be the first record in a transmission. This record
contains information about the sender and receiver, instruments, and
computer system whose records are being exchanged. It also identifies the
delimiter characters. The minimum information that must be sent in a Header
record is: H|\\^&{RT}
Example:
H|\\^&|||Afinion 2 Analyzer^^AF0000030|||||EPR||P|1|20100608185448|
"""
sender = ComponentField(
Component.build(
# H.5.1 Model name (always "Afinion 2 Analyzer")
TextField(name="name"),
NotUsedField(name="_"),
# H.5.3 DeviceID of measuring device
TextField(name="serial"),
))

# H.10.1 Name of the receiving application / dept.
receiver = TextField()
# H.12.1 Processing ID
processing_id = SetField(values=PROCESSING_IDS)
# H.13.1 ASTM-version used
version = TextField()


class PatientRecord(records.PatientRecord):
"""Patient Information Record (P)
This record is used to transfer patient information to the analyzer (test
order messages) or to the host (result messages).
Example:
P|1||43|||||U|
"""
# P.4.1 (local) patient ID
laboratory_id = TextField()


class OrderRecord(records.OrderRecord):
"""Order Record (O)
Example:
O|1||43|^^^CRP|||||||N||||^O||||||||^10124809||F|
"""
# O.4.1 Filler order number
instrument = IntegerField()

# O.5.4 Name of assay (e.g. CRP, ACR, Lipid Panel, HbA1C, ...)
test = ComponentField(
Component.build(
NotUsedField(name="_"),
NotUsedField(name="__"),
NotUsedField(name="___"),
TextField(name="name")
)
)

# O.12.1 Specimen action code
action_code = ConstantField(default="N")

# O.16.2 Specimen Source
biomaterial = ComponentField(
Component.build(
NotUsedField(name="_"),
SetField(name="source", values=SPECIMEN_SOURCES)
)
)


class CommentRecord(records.CommentRecord):
"""Comment Record (C)
"""


class ResultRecord(records.ResultRecord):
"""Record to transmit analytical data.
Examples:
R|1|^^^CRP|16|mg/L||||F||||20100608142352|
R|1|^^^ACR|<5.6|mg/g||<||F||||20100608140536|
R|1|^^^ACR|--- |mg/g||||F||||20100608140626|
R|1|^^^HbA1c|>15.0|%||>||F||||20201201142122|
"""

test = ComponentField(
Component.build(
NotUsedField("_"),
NotUsedField("__"),
NotUsedField("___"),
# R.3.4 Test Name (e.g. CRP, Alb, Creat, Trig, Chol, HbA1c, ..)
TextField("name")
)
)

# R.4.1 Measurement value
value = TextField()

# R.5.1 Units
units = TextField()

# R.7.1 Abnormal flags
# Precaution for results outside the measuring range:
# Calculated and measured results outside the measuring range are indicated
# with a comparator flag,">" or "<", passed along with a value in the
# observation value field. If the calculation is not possible, or the
# concentration can't be measured, the observation value field will contain
# "--- " instead of a value
abnormal_flag = SetField(values=ABNORMAL_FLAGS)

# R.9.1 Observation result status. Always "F" (final result)
status = ConstantField(default="F")

# R.11.1 Operator ID of the user, which the measurement has done
operator = TextField()

# R.13.1: Measurement time
completed_at = DateTimeField()


class RequestInformationRecord(records.RequestInformationRecord):
"""Request information Record (Q)
"""


class ManufacturerInfoRecord(records.ManufacturerInfoRecord):
"""Manufacturer Specific Records (M)
"""


class TerminatorRecord(records.TerminatorRecord):
"""Message Termination Record (L)
"""
1 change: 1 addition & 0 deletions src/senaite/astm/tests/data/abbott_afinion2.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
1H|\^&|||Afinion 2 Analyzer^^AF20052397|||||||P|1|20241206141235P|1||3643|||||UO|1||5|^^^HbA1c|||||||N||||^O||||||||^10228413||FR|1|^^^HbA1c|5.9|%||||F||3643||20241206140615L|1|NF2
Expand Down
135 changes: 135 additions & 0 deletions src/senaite/astm/tests/test_abbott_afinion2.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,135 @@
# -*- coding: utf-8 -*-

from unittest.mock import MagicMock
from unittest.mock import Mock

from senaite.astm import codec
from senaite.astm.constants import ACK
from senaite.astm.constants import ENQ
from senaite.astm.instruments import abbott_afinion2
from senaite.astm.protocol import ASTMProtocol
from senaite.astm.tests.base import ASTMTestBase
from senaite.astm.wrapper import Wrapper


class AbbottAfinion2(ASTMTestBase):
"""Test ASTM communication protocol for the Abbott Afinion™ 2 Analyzer, a
compact, rapid, multi-assay analyzer that provides valuable near patient
testing at the point of care.
"""

async def asyncSetUp(self):
self.protocol = ASTMProtocol()

# read instrument file
path = self.get_instrument_file_path("abbott_afinion2.txt")
self.lines = self.read_file_lines(path)

# Mock transport and protocol objects
self.transport = self.get_mock_transport()
self.protocol.transport = self.transport
self.mapping = abbott_afinion2.get_mapping()

def get_mock_transport(self, ip="127.0.0.1", port=12345):
transport = MagicMock()
transport.get_extra_info = Mock(return_value=(ip, port))
transport.write = MagicMock()
return transport

def test_communication(self):
"""Test common instrument communication """

# Establish the connection to build setup the environment
self.protocol.connection_made(self.transport)

# Send ENQ
self.protocol.data_received(ENQ)

for line in self.lines:
self.protocol.data_received(line)
# We expect an ACK as response
self.transport.write.assert_called_with(ACK)

def test_decode_messages(self):
self.test_communication()

data = {}
keys = []

for line in self.protocol.messages:
records = codec.decode(line)

self.assertTrue(isinstance(records, list), True)
self.assertTrue(len(records) > 0, True)

record = records[0]
rtype = record[0]
wrapper = self.mapping[rtype](*record)
data[rtype] = wrapper.to_dict()
keys.append(rtype)

for key in keys:
self.assertTrue(key in data)

def test_afinion2_header_record(self):
"""Test the Header Record wrapper
"""
wrapper = Wrapper(self.lines)
data = wrapper.to_dict()
record = data["H"][0]

# test sender name
self.assertEqual(record["sender"]["name"], "Afinion 2 Analyzer")
# test device id (serial)
self.assertEqual(record["sender"]["serial"], "AF20052397")

# test processing ID
self.assertEqual(record["processing_id"], "P")
# test astm version number
self.assertEqual(record["version"], "1")

def test_afinion2_patient_record(self):
"""Test the Patient Record wrapper
"""
wrapper = Wrapper(self.lines)
data = wrapper.to_dict()
record = data["P"][0]

# test (local) patient ID
self.assertEqual(record["laboratory_id"], "3643")

def test_afinion2_order_record(self):
"""Test the Order Record wrapper
"""
wrapper = Wrapper(self.lines)
data = wrapper.to_dict()
record = data["O"][0]

# test filler order number
self.assertEqual(record["instrument"], "5")

# test name of assay
self.assertEqual(record["test"]["name"], "HbA1c")

# test specimen action code
self.assertEqual(record["action_code"], "N")

# test specimen source
self.assertEqual(record["biomaterial"]["source"], "O")

def test_afinion2_result_records(self):
"""Test the Result Record wrapper
"""
wrapper = Wrapper(self.lines)
data = wrapper.to_dict()
records = data["R"]

# we should have 1 result
self.assertEqual(len(records), 1)

self.assertEqual(records[0]["test"]["name"], "HbA1c")
self.assertEqual(records[0]["value"], "5.9"),
self.assertEqual(records[0]["units"], "%")
self.assertEqual(records[0]["abnormal_flag"], None)
self.assertEqual(records[0]["status"], "F")
self.assertEqual(records[0]["operator"], "3643")

0 comments on commit 9fa2d6f

Please sign in to comment.