Purpose
Using this web service you can submit Diagnostic Report Package
Specification
Logic
Using this web service you can submit:
Object | Required | Part of Signed content |
---|---|---|
Observation | 0-* | true |
Diagnostic report | 1-1 | true |
Global and configurable parameters
https://e-health-ua.atlassian.net/wiki/spaces/EH/pages/907837561/Variables#Medical-Events
Request structure
See on Apiary
Example:
Authorize
Verify the validity of access token
in case of error return 401 ('Access denied')
Check user scope diagnostic_report:write in order to perform this action
in case of error generate 403 response ('Invalid scopes')
If BLOCK_UNVERIFIED_PARTY_USERS is true, then check party's data match following condition: verification_status != NOT_VERIFIED or (verification_status = NOT_VERIFIED and updated_at <= current_date - UNVERIFIED_PARTY_PERIOD_DAYS_ALLOWED):
in case not match - return 403 ("Access denied. Party is not verified")
Request to process the request using a token in the headers
Headers
Наприклад:
Content-Type:application/json
Authorization:Bearer mF_9.B5f-4.1JqM
Request data validation
Request validation
DS validation
DS must be valid
The document must be signed by recorder
Select PRM.parties.tax_id from PRM where PRM.parties.id ==( Select PRM.employees.party_id where PRM.employees.id==$.diagnostic_report.recorded_by.identifier.value)
ds.drfo == select a
in case of error retuert 409 "Document must be signed by the recorder of the diagnostic_report"
Recorder must be a current user
$.diagnostic_report.recorded_by.identifier.value is one of current user's employee
in case of error return 409 "Document must be sent by the recorder of the diagnostic_report"
Validate Legal Entity Type
Validate legal entity from token: legal_entities.type should be in me_allowed_transactions_le_types and legal_entities.status =='active'
Diagnostic report validation
Validate diagnostic_report _id is unique for MongoDB.Diagnostic_reports
Validate that diagnostic reports category corresponds to service category, that is references as code in DR
$.diagnostic_report.category[?]=PRM.services.category where PRM.services.id=$.diagnostic_report.code.identifier.value
in case of error return 422 "None of the diagnostic report categories matches with the service category"
Validate that service, referenced as a code in DR, is active
Note. For diagnostic_report.code pass only "service", and the "service_group" does not passPRM.services.is_active = true where PRM.services.id=$.diagnostic_report.code.identifier.value
in case of errror return 422 "Service is not active"
If service_requests.code.identifier.value is service, validate $diagnostic_report.code.identifier.value = service_requests.code.identifier.value
in case error return 409, "Service in diagnostic_report differ from service in service request"
if service_requests.code.identifier.value is service_group, validate $diagnostic_report.code.identifier.value in (SELECT service_id from service_inclusions where service_group_id='service_requests.code.identifier.value')
in case error return 409, "Service in diagnostic_report differ from services in service request's service_group"
Validate referrals
As a referral it can be referenced electronic (registered in the system) OR paper service request
Validate ($.diagnostic_report.based_on OR $.diagnostic_report.paper_referral) or none in request
Validate based on as Reference
Validate paper referral as Object (paper_referral)
Validate that service_request, referenced as based_on, is
in status is active or program_processing_status=in_progress (any status is valid in case program_processing_status= in_progress)
in case of error return 409 "Invalid service request status"
used_by_legal_entity==token.client_id OR null
in case of error return 409 "Service request is used by another legal_entity"
if program is defined than used_by_legal_entity==token.client_id or NULL
in case of error return 409 "Service request is used by another legal_entity"
if program is defined than program_processing_status == new, in_queue or in_progress
check that service_request contains based_on parameter
in case based_on present in service_request
verify care_plan:
It should be in active status
Care plan's period end (if exist) should be greater than current date or equal.
verify activity:
It has activity.detail.kind=service_request; activity.detail.product_reference=service_id.
It has scheduled, in_progress status
For an old activities (has no units in quantity) validate remaining_quantity as described at Submit Encounter Data Package: Related care plan validation
in case based_on not present in request skip previous validations.
Check if service request quantity is not exhausted as described at Submit Encounter Data Package: Related service request validation
Validate effective_period as a period (PeriodValidation)
Validate that issued is within acceptable limits
$.diagnostic_reports[*].issued<= current date_time
in case of error 422 "Issued date must be in past"
$.diagnostic_reports[*].issued>=current_date-diagnostic_report_max_days_passed
in case of error 422 "Issued must be greater than {{current_date-diagnostic_report_max_days_passed}}"
Validate recorded_by as Employee:
recorded_by employee_types allowed: DOCTOR, SPECIALIST, ASSISTANT, LABORANT
in case of error 422 "Invalid employee type"
Validate performer.identifier as Employee:
check that performer.identifier field is mandatory and employee_types allowed: DOCTOR, SPECIALIST, ASSISTANT, LABORANT
in case of error 422 "Invalid employee type"
Validate managing_organization is a current legal_entity
$.managing_organization.identifier.value==token.client_id
in case of error 409 "Managing organization does not correspond to user's legal entity."
Validate results_interpreter.identifier as Employee:
if the category.coding.code is in (
imaging
,diagnostic_procedure
) check that results_interpreter.identifier is an employee with employee_type = DOCTOR or SPECIALISTin case of error 422 "Invalid employee type"
if the category.coding.code is in (
diagnostic_curative
,laboratory_procedure
) - results_interpreter field is optional and the results_interpreter.identifier validation skipped
Validate only One of the fields is filled:
$.diagnostic_report[*].results_interpreter.reference OR $.diagnostic_report[*].results_interpreter.text
Validate division
Validate division as Reference(Referencevalidation)
$division is an existing ID in PRM.divisions
in case return 422, "Division with such id is not found"
division.status=ACTIVE and is_active=true
in case error return 409, "Division is not active"
division.legal_entity_id = $client_id
in case error return 409, "Division is not in current legal_entity"
Patient must be active
check that patients.status == active
in case patients.status == inactive
check mpi.persons(or mpi.prepersons if patients.preperson == true).updated_at and validate that now() - updated_at<=SUBMIT_DIAGNOSTIC_REPORT_PACKAGE_ALLOWED_PERIOD_MINUTES configuration (in minutes)
in case of error return 409 (Person is not active more that the allowed time for data submitting)
Validate patient verification status:
If diagnostic report has based_on with valid and active service request, then skip this validation.
Else check patient's verification_status is not equal to NOT_VERIFIED.
in case of error return 409, "Patient is not verified"
Validate for each element in specimens array, if submitted:
Check the element is a Reference on a specimen resource
in case of error return 422 “not allowed in enum”
Check it exists in DB and belongs to the same patient:
in case of error return 422 "Specimen not found"
Check status is available
in case of error return 422 "Invalid specimen status"
Observation validation
Several approaches are used for validating observations. This is driven by the natural growth of the system, which implies increased complexity and the different requirements for various groups, categories of observations and even the contexts in which they (observations) were created.
Currently, the following approaches are used:
Basic validations for all observations
Validation of observations for the encounter type "patient_identity".
A set of validations for rehabilitation observations based on ICF (International Classification of Functioning, Disability, and Health) dictionaries.
Validation of observations based on a configuration set defined for a specific observation code (new).
Basic validation and REHAB
Validate observations ids as primary keys (Primarykeyvalidation)
Validate that diagnostic_report of ALL observation is a current DR
$.observations[*].diagnostic_report.identifier.value==$.diagnostic_report.id
in case of error return "Submitted diagnostic report is not allowed for the observation"
$.observations[*].effective_period validate as PeriodValidation
Validate that the date is within acceptable limits
$.observations[*].issued <= current_time
in case of error return "Issued date must be in past"
$.observations[*].issued>=current_date-observation_max_days_passed
in case of error "Issued must be greater than {{current_date-observation_max_days_passed}}"
Validate performer as Employee:
performer employee_types allowed: DOCTOR, SPECIALIST, ASSISTANT, LABORANT
in case of error 422 "Invalid employee type"
Validate $.observations[*].value_period as a Period
Validate component
if $.observations[*].code.coding[*].system is "eHealth/ICF/classifiers":
Check number of component items with $.observations[*].components[*].code.coding[*].system = "eHealth/ICF/qualifiers":
if $.observations[*].code.coding[*].code starts from "b" letter, then components must contain exact 1 item with:
$.observations[*].components[*].code.coding[*].code = "extent_or_magnitude_of_impairment"
in case of missing qualifier - return 422 "Missing components with qualifiers <list of missing qualifiers>"
in case of incorrect number of qualifiers - return 422 "Required 1 component, but got <number of items>"
in case value is not active in dictionary - return 422 "Value is not active"
if $.observations[*].code.coding[*].code starts from "s" letter, then components must contain exact 3 items with:
$.observations[*].components[*].code.coding[*].code = "extent_or_magnitude_of_impairment"
$.observations[*].components[*].code.coding[*].code = "nature_of_change_in_body_structure"
$.observations[*].components[*].code.coding[*].code = "anatomical_localization"
in case of missing qualifier - return 422 "Missing components with qualifiers <list of missing qualifiers>"
in case of incorrect number of qualifiers - return 422 "Required 3 components, but got <number of items>"
in case value is not active in dictionary - return 422 "Value is not active"
if $.observations[*].code.coding[*].code starts from "d" letter, then components must contain exact 2 items with:
$.observations[*].components[*].code.coding[*].code = "performance"
$.observations[*].components[*].code.coding[*].code = "capacity"
in case of missing qualifier - return 422 "Missing components with qualifiers <list of missing qualifiers>"
in case of incorrect number of qualifiers - return 422 "Required 2 components, but got <number of items>"
in case value is not active in dictionary - return 422 "Value is not active"
if $.observations[*].code.coding[*].code starts from "e" letter, then components must contain exact 1 item with:
$.observations[*].components[*].code.coding[*].code = "barrier_or_facilitator"
in case of missing qualifier - return 422 "Missing components with qualifiers <list of missing qualifiers>"
in case of incorrect number of qualifiers - return 422 "Required 1 component, but got <number of items>"
in case value is not active in dictionary - return 422 "Value is not active"
Check $.observations[*].components[*].value_codeable_concept is present in each item with $.observations[*].components[*].code.coding[*].system = "eHealth/ICF/qualifiers" and $.observations[*].components[*].value_codeable_concept.coding[*].system corresponds to $.observations[*].components[*].code.coding[*].code
in case not present or doesn't correspond return 422 "Doesn't correspond to $.observations[i].components[i].code"
in case code value is not active in dictionary - return 422 "Value is not active"
$.observations[*].components[*].value_period as a Period
Validate $.observations.code
if observations.code.coding[*].code value is included in chart variables 'OBSERVATION_CODES_WITH_<VALUE_TYPE>_REQUIRED', <value_type> field is mandatory
in case of error return 422 “This field is required for code = <code>“
if observations[*].code.coding[*].system is "eHealth/ICF/classifiers", then:
Check observations[*].categories[*].coding[*].system should contain "eHealth/ICF/observation_categories"
in case of error return 422 “Code doesn't match observation category“
Check observations[*].components are presented
in case of error return 422 “Components required“
Validate $.observations[*].categories
is an array with only one item
in case of error return 422 "Expected a maximum of 1 items but got <number of elements>"
$.observations[*].categories[*].coding[*].system is "eHealth/observation_categories" or "eHealth/ICF/observation_categories"
in case of error return 422 "Value is not allowed in enum"
$.observations[*].categories[*].coding[*].code corresponds to the system
in case of error return 422 "Value is not allowed in enum"
If $.observations[*].categories[*].coding[*].system = "eHealth/ICF/observation_categories", then $.observations[*].code should be filled from "eHealth/ICF/classifiers" dictionary.
in case of error return 422 “Code doesn't match observation category“
Validate specimen, if submitted:
Check the field is a Reference on a specimen resource
in case of error return 422 “not allowed in enum”
Check it exists in DB and belongs to the same patient:
in case of error return 422 "Specimen not found"
Check status is available
in case of error return 422 "Invalid specimen status"
Validate observations using configuration set
This validations must be done on top of Basic validations and only in case there is defined configuration set for such observation
Search for active configuration set (is_active = true) using observation code (code + system) and perform additional validation. If there is no configuration for such code then this additional validations must be skipped
Category
Check that observation.categories.coding.system and observation.categories.coding.code matches with one of the options in CATEGORY configuration parameter
Components
Check that all components in observation.components are unique and not duplicated if COMPONENTS_UNIQUE is set to true. Otherwise skip this step
in case of error return 422 “Observation components must be unique“
Check that all required components are present in observation.components (declared in COMPONENTS_REQUIRED)
values in COMPONENTS_REQUIRED.check means all required and observation components must include all of them
if COMPONENTS_REQUIRED is missing or is empty then skip this validation
in case of error return 422 “Not all required components provided for this observation code“
Check that there are no items in observation.components other than those allowed, which are defined by the COMPONENTS_OPTIONAL
values in COMPONENTS_OPTIONAL.check means all allowed (except COMPONENTS_REQUIRED) observation components that could be used for this observation code
if COMPONENTS_OPTIONAL is missing or is empty then no additional components are allowed
in case of error return 422 “Component code is not allowed for this observation code“
Check that component value type matches with COMPONENTS_TYPE configuration parameter
search for related configuration in COMPONENTS_TYPE.condition using observation.component.code.coding.system and observation.component.code.coding.code
check that observation.component.value[x] type matches with the type declared in COMPONENTS_TYPE.check
if COMPONENTS_TYPE is missing or is empty or there is no matched configuration then skip this validation
in case of error return 422 “type mismatch. Expected {COMPONENTS_TYPE.check} but got {entered type}“
Check that dictionary used to represent component with value type codeable_concept matches with declared in COMPONENTS_BINDING configuration parameter
Must be done only in case when component value type is codeable_concept (observation.component.value[x] type is codeable_concept)
search for related configuration in COMPONENTS_BINDING.condition using observation.component.code.coding.system and observation.component.code.coding.code
check that observation.component.value_codeable_concept.coding.system matches with the system (dictionary) declared in COMPONENTS_BINDING.check
if COMPONENTS_BINDING is missing or is empty or there is no matched configuration then skip this validation
in case of error return 422 “Incorrect dictionary. Expected {COMPONENTS_TYPE.check}“
Check that code system and value used to represent quantity in component are allowed according to COMPONENTS_QUANTITY_CODES configuration parameter
Must be done only in case when component value type is quantity (observation.component.value[x] type is quantity)
search for related configuration in COMPONENTS_QUANTITY_CODES.condition using observation.component.code.coding.system and observation.component.code.coding.code
check that observation.component.value_quantity.coding.system and observation.component.value_quantity.coding.code matches respectively with one of configurations in COMPONENTS_QUANTITY_CODES.check
if COMPONENTS_QUANTITY_CODES is missing or is empty or there is no matched configuration then skip this validation
in case of error return 422 “Such quantity system and quantity code are not allowed in this component“
Check that component value is within the declared in COMPONENTS_QUANTITY_BOUNDARIES configuration parameter boundaries
Must be done only in case when component value type is quantity (observation.component.value[x] type is quantity)
search for related configuration in COMPONENTS_QUANTITY_BOUNDARIES.condition using (observation.component.code.coding.system and observation.component.code.coding.code for COMPONENTS_QUANTITY_BOUNDARIES.condition.component_code) and (observation.component.value_quantity.system and observation.component.value_quantity.code for COMPONENTS_QUANTITY_BOUNDARIES.condition.quantity_code)
check that configuration.check.min <= observation.component.value_quantity.value >= configuration.check.max
if COMPONENTS_QUANTITY_BOUNDARIES is missing or is empty or there is no matched configuration then skip this validation
in case of error return 422 “Incorrect value. Expected value to be in range {configuration.check.min} and {configuration.check.max}“
Check that only allowed extensions are used in components (declared in COMPONENTS_ALLOWED_EXTENSIONS)
values not declared in COMPONENTS_ALLOWED_EXTENSIONS.check is prohibited and is not allowed in observation.components
check that observation.component.value_codeable_concept.coding.extension contains only extensions declared in COMPONENTS_ALLOWED_EXTENSIONS.check
in case of error return 422 “Extension {observation.component.value_codeable_concept.coding.extension} is not allowed“
check that there are no duplicated extensions in observation.component.value_codeable_concept.coding.extension
in case of error return 422 “Duplicated extension“
check that extension is valid according to https://e-health-ua.atlassian.net/wiki/spaces/REHABILIT/pages/18431246564/UPD+Medical+Events+MongoDB+Data+Model#Predefined-extensions
in case of error return 422 “Invalid extension“
Check that extension value is correct according to COMPONENT_SCORE configuration parameter
Must be done only in case when component value type is codeable_concept (observation.component.value[x] type is codeable_concept)
search for the score associated with provided value (declared in COMPONENT_SCORE configuration parameter) using component value (observation.component.value_codeable_concept.coding.system and observation.component.value_codeable_concept.coding.code) and component code (observation.component.code.coding.system and observation.component.code.coding.code).
check that extension value (observation.component.value_codeable_concept.coding.extension(code = item_weight).value_decimal) is equal value declared in COMPONENT_SCORE.check for such component
in case of error return 422 “Incorrect component score“
Value
Check that observation value type matches with RESULT_TYPE configuration parameter
check that observation.value[x] type matches with the type declared in RESULT_TYPE.check
if RESULT_TYPE is missing or is empty then skip this validation
in case of error return 422 “type mismatch. Expected {RESULT_TYPE.check} but got {entered type}“
Check that dictionary used to represent observation value matches with declared in RESULT_BINDING configuration parameter
Must be done only in case when observation value type is codeable_concept (observation.value[x] type is codeable_concept)
check that observation.value_codeable_concept.coding.system matches with the system (dictionary) declared in RESULT_BINDING.check
if RESULT_BINDING is missing or is empty then skip this validation
in case of error return 422 “Incorrect dictionary. Expected {RESULT_BINDING.check}“
Check that code system and value used to represent quantity in observation are allowed according to RESULT_QUANTITY_CODES configuration parameter
Must be done only in case when component value type is quantity (observation.value[x] type is quantity)
check that observation.value_quantity.coding.system and observation.value_quantity.coding.code matches respectively with one of configurations in RESULT_QUANTITY_CODES.check
if RESULT_QUANTITY_CODES is missing or is empty or there is no matched configuration then skip this validation
in case of error return 422 “Such quantity system and quantity code are not allowed in observation result“
Check that observation value is within the declared in RESULT_BOUNDARIES configuration parameter boundaries
Must be done only in case when observation value type is quantity (observation.value[x] type is quantity)
search for related configuration in RESULT_BOUNDARIES.condition using observation.value_quantity.coding.system and observation.value_quantity.coding.code
check that configuration.check.min <= observation.value_quantity.value >= configuration.check.max
if RESULT_BOUNDARIES is missing or is empty or there is no matched configuration then skip this validation
in case of error return 422 “Incorrect value. Expected value to be in range {RESULT_BOUNDARIES.check.min} and {RESULT_BOUNDARIES.check.max}“
Check that observation value calculated correctly according to the method declared in RESULT_SCORE_CALCULATION configuration parameter
There are different supported methods to calculate and control result (“SUM”)
If `SUM` specified
go through all observation.components
sum all scores provided in item_weight extension observation.component.value_codeable_concept.coding.extension(code = item_weight).value_decimal and all components where component value type is quantity and quantity system = eHealth/ucum/units and quantity code = ScoreOf
ensure that this result is equal to observation.component.value_quantity.value
if RESULT_SCORE_CALCULATION is missing or is empty then skip this validation
in case of error return 422 “Incorrect observation result“
Processing
API paragraph not found
Response structure
See on Apiary
Example:
Post-processing processes
Set managing_organization for submitted observations
ME.conditions{cond_id}.managing_organization=token.client_id.
in case service request has reference on care plan and diagnostic report.based_on was filled set $.diagnostic_report.id to related to $.service_request $.activity[].outcome_reference
Update $ .activity.status to in_progress if previous activity status was scheduled
Update the $ .activity.remaining_quantity parameter by the value achieved at Submit Encounter Data Package: Related care plan validation (for an old activities only that has no quantity units)
Update remaining_quantity in the service request with a value that was calculated in the Submit Encounter Data Package: Related service request validation
Set origin_episode (if $.based_on in request and $.based_on.service_request.context != null) for Diagnostic Report:
origin_episode is an object from the $.service_request.context.episode