- Overview
- Sections of a pathway
- Steps
- Use Patient
- Delay
- Admission
- Transfer
- Transfer in Error
- Document
- Discharge
- Discharge in Error
- Registration
- Cancel Admit/ Visit
- Cancel Transfer
- Cancel Discharge
- Add Person
- Update Person
- Pre Admission
- Pending Admission
- Pending Transfer
- Pending Discharge
- Delete Visit
- Cancel Pending Admission
- Cancel Pending Transfer
- Cancel Pending Discharge
- Order
- Results
- Merge
- Bed swap
- Track Departure / Track Arrival
- AutoGenerate
- Clinical Note
- Hardcoded message
- Generic
- GenerateResources
- Order profiles
- Explicitly specify results for each test type in the order profile (recommended)
- Only specify results for a subset of test types
- Explicitly specify result as random
- Do not specify results - implicitly random
- Results for a random order profile
- Results for non-existing order profile
- Results with an unknown name for an existing order profile
- Default values for Order Profiles
- Pathway with multiple Orders and Results
- Pathway with Result but no Order
- Pathways with multiple Results with the same order_id
- Step parameters
- Allergies
- Locations
- Appendix
This page explains how to write pathways to be run by Simulated Hospital.
A pathway is a sequence of events that happen to patients in a hospital. Simulated Hospital generates patients that follow pathways. Pathway definitions can be written in YAML or JSON format.
At startup, Simulated Hospital loads pathways from the directory specified by
the pathways_dir
command-line argument. All files
inside that directory are read and loaded. Each file can contain one or more
pathways. All pathways defined in a single file must use the same format (either
YAML or JSON, but not a combination of both). All pathways need to be valid
pathways; if not, Simulated Hospital won't start.
The examples in this guide use YAML for brevity.
A pathway consists of the following parts:
pathway_name: # The unique name of this pathway.
persons: # A description of the person or persons the pathway relates to.
...
consultant: # The consultant to use whenever it is needed in the pathway.
...
percentage_of_patients: # The percentage of patients the pathway relates to.
...
historical_data: # Events that happened in the past.
...
pathway: # Actual events in the pathway.
...
All sections are optional, but either historical_data
or pathway
, or both,
must be present.
Events are also referred to as steps.
The persons
section configures the person or persons the pathway refers to.
This data is used to construct PID segments.
The persons
section is optional. If the persons section is not set, the data
about the patient is generated randomly. This applies to individual fields too:
if a pathway only specifies a subset of fields, the rest of them will be
generated randomly. The how to configure data
page explains how data about persons is generated and how you can change it.
See an example of a persons
section with a single patient main_patient
that
sets the MRN, first name, surname, age and postcode:
sample_pathway:
persons:
main_patient:
mrn: "291847192817319"
first_name: "Elizabeth"
surname: "Smith"
age:
from: 20
to: 25
address:
postcode: "SW1 4DN"
...
A persons section can refer to multiple patients. This is useful for pathways
where two medical records interact, such as pathways that do bed swaps or
patient merges. See an example of a persons
section with two patients:
sample_pathway:
persons:
main_patient:
first_name: "Main Patient"
unconscious_patient:
first_name: "Unconscious"
...
The steps in the Pathway section can refer to each of these persons
by their identifiers (main_patient
or unconscious_patient
). See
Use Patient for an example.
The following list contains the fields that can be configured in a pathway. Remember that if a field is not specified, it is generated randomly.
nhs
: NHS number (National Health System number). Simulated Hospital generates valid NHS numbers only, but there is no validation when the NHS number is specified directly in a pathway.mrn
: Medical Record Number (MRN). By default this is a random 32-bit integer. See how to extend Simulated Hospital to learn how to configure this.age
: the age of the patient in years. You can specify a range using thefrom
andto
attributes, and the specific day in a year the person was born in theday_of_year
attribute.date_of_birth
: the specific date of birth for the patient. Only one ofage
ordate_of_birth
can be set.gender
: the gender of the person.first_name
: the first name of the person. Names and surnames allow patients to be easily found when searching for examples of specific situations, and therefore many of the pathways provided by default set the first name of the patients to something that indicates the pathway the patient is in, e.g. AKI Scenario 1.surname
: the surname of the person.address
has the following subfields:first_line
second_line
city
postcode
country
: the country code, e.g. GBtype
is the type of address, e.g. HOME or WORKall_random
is a "true" or "false" value that indicates whether to populate all fields with randomly generated values.
The following fields have special generation rules:
- Name. When the name not provided, it is chosen based on the gender and the date of birth.
- Address. Every field in the address can be overridden separately, except
second_line
. Iffirst_line
is set,second_line
will be re-generated.
The following demographics are always generated randomly:
- Middle name is generated based on gender. It can be empty.
- Ethnicity: by default it is based on the ethnicity frequency in London from wikipedia.
- Patient Class: based on the patient class and type distribution and modified according to the steps, i.e if the patient is admitted their class is set to “INPATIENT”.
- Prefix: based on gender.
- Suffix and Degree can be empty.
- Telephone number.
The consultant
section configures the consultant that will be used whenever a
consultant is needed in the pathway. This data is used to populate information
about clinicians in EVN, OBR, OBX, PV1, DG1 and PR1 segments.
Diagnoses and procedures always use an arbitrary consultant to indicate that past diagnoses and procedures are generally done by other clinicians.
The consultant
section is optional. If the consultant section is not set, the
consultant is randomly chosen from the set of doctors defined in the
[doctors_file
argument]./arguments.md#data-configuration]. To use an existing
predefined doctor, set the id
field only, for example:
consultant:
id: C12345678
The identifier "C12345678" must match an pre-defined doctor.
To use a doctor that is not part of the pre-loaded pool of doctors, set all fields, for example:
consultant:
id: C12345678
prefix: Dr.
first_name: John
surname: Smith
...
This new doctor will be added to the pool of doctors, and pathways can now refer
to it by the id
only in the current run of Simulated Hospital. Simulated
Hospital sets the specialty from the hospital_service
field specified in the
HL7 message configuration file.
The percentage_of_patients
is a floating number that represents the percentage
of patients the given pathway relates to. This number will determine the
likelihood of the pathway being run in comparison to other pathways. The greater
the value, the more often this pathway will be run.
The percentage_of_patients
values for all pathways should sum up to 100. If
the pathway does not have percentage_of_patients
specified, its
percentage_of_patients
will be assigned from the remaining budget, ie: if the
subset of the pathways have the percentage_of_patients
specified and it sums
up to 70%, the remaining 30% would be split equally across all remaining
pathways.
There following flags can be set to control the percentage calculation:
max_significant_digits
- significant digits accepted forpercentage_of_patients
(default: 3)strict_percentage_validation
- whether to fail if the sum ofpercentage_of_patients
differs from 100 more thanpercentage_tolerance
(default: false)percentage_tolerance
- How much the accumulatedpercentage_of_patients
can deviate from 100 (default: 10)default_percentage
- Defaultpercentage_of_patients
for pathways that don't specify it (default: 1). Note: this is only used if it is impossible to calculate the percentage by splitting the remaining budget across pathways that do not have the percentage specified.
The historical_data
section contains the list of events that happened in the
past and might be relevant to the patient's current pathway, for instance, it
may contain historical test results in the patient's medical history. These
events are performed before the actual pathway starts.
Any supported events, except Delay
(see Steps), can be included in
this section. All events in this section must have a negative time_from_now
,
that represents when in the past the event took place, see the following
example:
pathway_name:
historical_data:
- result:
order_profile: UREA AND ELECTROLYTES
results:
- test_name: Creatinine
value: 126.00
unit: UMOLL
abnormal_flag: HIGH
parameters:
time_from_now: -8783h
The pathway
section contains the actual events in the pathway. Any supported
events can be included in this section.
The following is an example of a pathway that performs an admission, a test result, and a discharge of a patient, with a few delays in between them.
aki_scenario_1:
pathway:
- admission:
loc: Renal
- delay:
from: 10m
to: 30m
- result:
order_profile: UREA AND ELECTROLYTES
results:
- test_name: Creatinine
value: 254.00
unit: UMOLL
abnormal_flag: HIGH
- delay:
from: 30m
to: 60m
- discharge: {}
Simulated Hospital supports multiple step types that refer to different events
at a hospital, for instance, admissions, discharges, bed swaps or test results.
Each step type translates to a different HL7 message. For instance, an
admission
step generates an ADT A01 message.
The Messages types and pathway events
table connects message types and the steps that generate them.
The use_patient
step defines which patient should be used from now on in the
pathway. The patient can be referred to by the person identifier specified in
the Persons section in the pathway, or by their MRN.
sample_pathway:
persons:
main_patient:
first_name: "Main Patient"
unconscious_patient:
first_name: "Unconscious"
pathway:
- use_patient:
patient: unconscious_patient
The delay
step adds a delay between 2 steps. Delay steps represent delays
between events in hospitals and enable building more realistic pathways. Delay
steps don't generate messages.
The duration specified in a Delay step runs in real time. For instance, the following steps generate a random delay between an admission and a discharge of 30 to 60 minutes:
sample_pathway:
pathway:
- admission:
loc: Renal
- delay:
from: 30m
to: 60m
- discharge: {}
When the pathway runs, Simulated Hospital Patients will wait for 30 to 60 minutes before issuing the discharge after the admission.
An admission
step represents a hospital admission and produces an A01 message.
This step type requires the name of the point of care (loc
) the patient is
being admitted to, for example:
admission_only:
pathway:
- admission:
loc: Renal
Optionally you can specify an admit reason for the patient. This will add the admission reason to the PV2.3 Admit Reason field, for example:
admission_with_reason:
pathway:
- admission:
loc: Renal
admit_reason: Kidney problems
The point of care must be one of the locations available in the hospital, see the Locations section.
An admission does the following:
- Generate a location in the given point of care
- Generate a visit identifier
- Set the admission date to the current time
- Set the patient class to INPATIENT
A transfer
step represents a transfer from one location to another and
produces an A02 message. This step requires the name of the point of care
(loc
) the patient will be transferred to. If the patient is currently
occupying a bed somewhere else, such bed will be marked as free, and a new bed
will be assigned in the new point of care. Example:
pathway_with_transfer:
pathway:
- admission:
loc: Non-renal
- transfer:
loc: Renal
When the pathway runs, the patient is admitted to the Non-renal
ward and then
transferred to the Renal
ward.
A transfer_in_error
step behaves like a Transfer step but does
not free the current bed or allocate a new bed. This step represents a transfer
that was requested in error and will be cancelled with a
Cancel Transfer step. Since the transfer is going to be
cancelled, Simulated Hospital Patients does not need to do any space management.
A document
step creates documents such as an autopsy reports, diagnostic
images, discharge summaries and cardiodiagnostics. It produces a MDM^T02 message
with an TXA and an OBX segment.
This step allows customizing the document properties and/or the content.
This step has the following optional parameters to set the document properties:
document_type
populates the TXA.2-Document Type field. If not set in the pathway, the document type is chosen at random from the configurable list in the hl7 message configuration file.completion_status
populates the TXA.17-Document Completion Status field. If not set in the pathway, it defaults to DO (Documented).observation_identifier_id
populates the ID of the OBX.3-Observation Identifier field. If not set in the pathway, it defaults to Established Patient 15.observation_identifier_text
populates the Text of the OBX.3-Observation Identifier field. If not set in the pathway, it defaults to Established Patient 15.observation_identifier_coding_system
populates the Coding System of the OBX.3-Observation Identifier field. If not set in the pathway, it defaults to Simulation.
Example of a new document:
pathway_with_document:
pathway:
- document:
document_type: DS
completion_status: IP
observation_identifier_id: obs-id
observation_identifier_text: obs-text
observation_identifier_coding_system: coding-system
The document
step generates random textual content divided into lines, and
each line of the content becomes an OBX segment.
It is possible to set fixed content for the header, ending, or both, in order to make the document follow the expected structure of a clinical system.
The following optional fields control the document's content:
header_content_lines
sets the first lines in the document if fixed content needs to be set.ending_content_lines
sets the last lines in the document if fixed content needs to be set.num_random_content_lines
specifies the range of content lines using thefrom
andto
attributes. If not set in the pathway, Simulated Hospital generates a random number of lines between 10 and 50. If you do not want random content, set this to 0 or set as empty.
Example of a document with a fixed header and a fixed ending, and random content in between:
- document:
header_content_lines:
- header-text-line-1
- header-text-line-2
ending_content_lines:
- ending-text-line-1
- ending-text-line-2
Example of a document with fixed content only:
- document:
header_content_lines:
- content-text-line-1
- content-text-line-2
- content-text-line-3
num_random_content_lines: {}
The document
step includes an option to update the content lines for an
existing document given a pathway Document ID and an update type to be one of
the following:
- append
- overwrite
The following required fields control the document update:
id
is used to link documents within a pathway. This should be set when generating a new document if there is intention to update this document. This does not set field TXA.12-Document_ID, which is always randomly generated.update_type
sets the type of update to perform.
Example of a document update of type append
:
pathway_with_document:
pathway:
- document:
id: doc-id1
- document:
id: doc-id1
update_type: "append"
ending_content_lines:
- ending-text-line-1
This pathway generates two messages for the same document. In the first message, the document will have between 10 and 50 OBX segments with randomly generated content. The second message will have an additional OBX with the content "ending-text-line-1" after the previous randomly generated content.
Example of a document update of type overwrite
:
pathway_with_document:
pathway:
- document:
id: doc-id1
- document:
id: doc-id1
update_type: "overwrite"
header_content_lines:
- header-text-line-1
num_content_lines:
from: 10
to: 10
This pathway generated two messages for the same document. In the first message, the document will have between 10 and 50 OBX segments with randomly generated content. The second message will replace these segments with an OBX with the content "header-text-line-1" followed by 10 OBX segments with randomly generated content.
A discharge
step represents a discharge and produces an A03 message. This step
does not have parameters. A discharge step does the following:
- Set the patient's discharge date
- Set the patient class to OUTPATIENT
- Mark the patient's bed as available
A discharge_in_error
step behaves as a Discharge step but does
not mark the patient bed as available. This step represents a discharge that was
issued in error and will be cancelled with a
Cancel Discharge.
This step does not have parameters.
A registration
registers a patient and produces an A04 message. An optional
patient_class
field sets a specific patient class, see example:
pathway_with_registration:
pathway:
- registration:
patient_class: EMERGENCY
If the patient_class
is set, it will be used for both the Patient Class
(PV1.2) and Patient Type (PV1.18). If no patient_class
field is set, the class
is generated randomly based on the distribution from the patient_class.csv
file.
A cancel_visit
step cancels the latest admission or visit and produces a A11
message.
This step takes no parameters.
A cancel_visit
step removes the cancelled admission or visit from the
patient's history.
A cancel_transfer
cancels a transfer and produces an A12 message.
Use Transfer in error before cancel_transfer
so that the
patient location is not marked as available between the erronous transfer and
the cancellation.
This step takes no parameters.
A cancel_discharge
step cancels the latest discharge and sends an A13 message.
Use Discharge in error before cancel_discharge
so that
the patient location is not marked as available between the erronous discharge
and the cancellation.
This step takes no parameters.
An add_person
creates a person and sends an A28 message.
It uses the person specified in the pathway. It is only allowed as the first step of the pathway.
An update_person
updates an existing person. It sends an A08 message (Update
patient information) if the person is an inpatient, and an A31 (Update person
information) if the person is not an inpatient.
update_person
might contain a valid person as the parameter. If it contains a
person, the current patient’s details will be updated with those of the person.
A person inside an update_person
can have some fields set to an automatic
random value (for instance, the address and/or the surname), which updates the
patient’s details with new random values. This is particularly useful when we
want those values to be updated but we don’t want to set fixed values for all
the patients in such pathway.
To set a random value for a surname, use the keyword RANDOM. For addresses,
every individual field inside an address can also be set to RANDOM.
Alternatively, you can use all_random: true
to generate a new random full
address.
If an update_person
changes or sets the gender, the middlename and prefix will
be generated again only if the new gender is different than the previous one.
An update_person
can also have a set of diagnoses and procedures specified.
If the code or description (or both) is set to RANDOM, a random diagnosis or
procedure is generated from the list of all valid diagnoses / procedures. The
time_from_now
can be omitted in this case - it will be randomly set as maximum
1 year from now.
If only code is set for the given diagnosis / procedure, and it matches one of the diagnosis / procedure loaded, the description is also populated. Same works the other way round (i.e., if only description is specified in the pathway).
Example:
- update_person:
person:
first_name: "Updated Person"
surname: RANDOM
address:
all_random: true
diagnoses:
- type: Working
code: A01.0
description: Typhoid fever
datetime:
time_from_now: -48h
procedures:
- code: RANDOM
datetime:
time_from_now: -24h
A pre_admission
generates a pre admission event and an A05 message. A
pre_admission
requires the following parameters:
loc
: The location where the patient will be admittedexpected_admission_time_from_now
: The duration of time until the admission takes place
The HL7 segments of A05 messages are syntactically identical to those of A14 (Pending Admission) but A05 implies that an account should be opened for the purposes of tests prior to the admission.
A pending_admission
generates a pending admission event and an A14 message. A
pending_admission
requires the following parameters:
loc
: The location where the patient will be admittedexpected_admission_time_from_now
: The duration of time until the admission takes place
A pending_transfer
generates a pending transfer event and an A15 message. A
pending_transfer
requires the following parameters:
loc
: The location where the patient will be transferredexpected_transfer_time_from_now
: The duration of time until the transfer takes place
A pending_discharge
generates a pending discharge event and an A16 message. A
pending_discharge
requires the following parameters:
expected_discharge_time_from_now
: The duration of time until the discharge takes place
A delete_visit
deletes a patient visit and sends a A23 message. The
delete_visit
step is used to delete a previously discharged or canceled visit
record. To cancel an ongoing visit, use cancel_visit
instead. In the current
implementation, we default to deleting the most recent discharged or canceled
visit, so no parameters are needed.
A cancel_pending_admission
cancels a pending admission and generates an A27
message. In the A27 message, PV2-1 field contains the location where the patient
is no longer being admitted to. A cancel_pending_admission
step must always be
preceded by a pending_admission
step. No parameters are required.
A cancel_pending_transfer
cancels a pending transfer and generates an A26
message. In the A26 message, the PV2-1 field contains the location where the
patient is no longer being transferred to. A cancel_pending_transfer
step must
always be preceded by a pending_transfer
step. No parameters are required.
A cancel_pending_discharge
cancels a pending discharge and generates an A25
message. A cancel_pending_discharge
step must always be preceded by a
pending_discharge
step. No parameters are required.
An order
event places an order and generates an ORM message.
An order
event has the following optional fields:
order_id
: a unique identifier for the order within the pathway. Order identifiers are used to linkorder
andresult
events within the same pathway, so that Simulated Hospital knows that those events relate do the same order.order_profile
: the type of order which populates the "OBR.4 - Universal Service Identifier" field. This can be either an order profile from the pre-loaded order profiles or a different value.order_status
: the status of the order to set in the "OBR.5 - Order Status". If not specified, Simulated Hospital uses theorder_status.in_process
value from the HL7 config.
At least one of order_id
or order_profile
needs to be present.
Simulated Hospital generates a Placer number ("ORC.4 - Placer Group Number") for each new order.
After an order
step generates an ORM message, by default it also generates an
ORR message 1-10 seconds afterwards. The ORR (O02) message acknowledges that the
ORM message was received.
The MSA segment of the ORR message contains the message control ID from the ORM message, connecting the message to the order it acknowledges.
The ORC segment of the ORR message is the same as in the ORM message, with the exception that the Order Control field is set to OK.
If you don't want an order acknowledgement to be sent after a particular order,
on the order
step you can set the boolean parameter
no_acknowledgement_message
to true
and the ORR message won’t be generated.
A results
step generates a set of results and an ORU message. Given an order
profile name and the list of results, the full list of results for each test
type for the Order Profile is generated. For each test type:
- Value and unit - if specified in the pathway, they are used, otherwise
generate random value from the normal range for test type. Use
EMPTY
to force an empty value. - Value type - if result value is numerical, set it to "NM", otherwise to
"TX". If the value is set to
EMPTY
, the value type defaults to the value specified in the Order Profile, or is set to an empty string if there is no matching Order Profile. - Reference range (
reference_range
) - if specified in the pathway, it is used, otherwise it is set to default reference range for the test type
This data is used to construct OBX segment.
Also, in this step the Filler number is generated and populated in the ORC and OBR segment, on top of Placer number, which is already generated in the Order step.
The times associated with a result:
- Collected
- Received in Lab
- Reported
are all included within the ORUs. The max time between each of them is specified by values set in the configuration. However the time differences are bounded by the delay between when the ORM and the ORU were fired.
In order to model some of the behaviour we find in messages received from
partners, we have made it possible to specify blank times in the pathway for
collected
and/or received_in_lab
as follows:
collected: EMPTY
received_in_lab: EMPTY
A results
event generates a HL7 message of type ORU^R01
by default. Use the
trigger_event
field to generate an ORU message with a different trigger event:
R01
(default if empty), R03
or R32
, all case insensitive. Any other value
will be considered invalid.
Evern item in the results
field creates a new OBX segment. You can specify
notes for each of the results, which will become NTE segments associated to the
OBX segment. Example:
- result:
order_id: creat1
order_profile: UREA AND ELECTROLYTES
results:
- test_name: Creatinine
value: 500.00
unit: UMOLL
reference_range: 49 - 92
abnormal_flag: DEFAULT
notes:
- "First note for the Creatinine"
- "Second note for the Creatinine"
In order to recreate the midnight case (a collected time of 00:00), set the following:
collected: MIDNIGHT
A merge
merges two or more patients.
This step requires you to specify in advance one (or more) valid MRNs to merge the current patient with, or you can define and use multiple Persons in the pathway and use their identifiers.
The Merge message will be of type ADT^A34
if there is only one MRN or
identifier in the children
field, and of typeADT^A40
if there are more or if
the (optional) field force_a40
is set to true
. Example:
merge:
parent: CURRENT
children: [1234]
force_a40: true
A bed_swap
performs a bed swap between two patients. It produces A17 message.
This step requires you to specify in advance one valid MRN or patient identifier to bed swap the current patient with. You can define and use multiple Persons in the pathway and use their identifiers.
Example of a bed swap:
bed_swap:
patient_1: CURRENT
patient_2: 1234
Note that only one MRN can be set here, as opposed to Merge steps that allow a list of MRNs.
Track Departure and Track Arrival events represent changes in a patient's physical location (inpatient or outpatient) that do not change the official census bed location. See http://www.hl7.eu/refactored/msgADT_A09.html and http://www.hl7.eu/refactored/msgADT_A10.html.
A track_departure
event generates an A09 message. The event is triggered when
there is about to be a change in the patient’s location, but an official A02
transfer hasn’t been issued. The patient could be leaving the floor or the
building, but must stay within the same healthcare institution. There are three
modes in which this event can occur. For two of these modes, the event must be
followed by a track_arrival
event that completes the move, otherwise the
patient location will remain in an inconsistent state.
-
transit mode: the patient is departing to a non-temporary location and there is a gap between departure and arrival times. Use the optional field
destination_bed
to specify the specific bed the patient is departing to. Add atrack_arrival
event withtransit
mode to complete the move. Theloc
field must be the same for both events. -
temporary mode: the patient is moving to a temporary location (e.g., Hallway, O/R, X-RAY, Limbo). Add a
track_arrival
event withtemporary
mode to complete the move.NOTE: multiple
track_departure
events in this mode can be issued before atrack_arrival
event, e.g. the patient could be sent to Hallway and then X-ray before enough time has passed to issue atrack_arrival
event for the Hallway location. -
track mode: patient is departing to a new non-temporary location. A
track_arrival
event is not needed after this. Use the optional fielddestination_bed
to set the specific bed the patient is departing to.
A track_arrival
event generates an A10 message. There are multiple modes of
track arrival events that are similar to the track_departure
events:
-
transit mode: the patient has arrived to a non-temporary location. This event must follow a
track_departure
event withtransit
mode. -
temporary mode: the patient has arrived from a temporary location. The location the patient has arrived to can be temporary or permanent. If the location of arrival is temporary, set the
is_temporary
field. This field can only be set intemporary
mode. If the location is permanent, use the optional fieldbed
to set the specific bed the patient has arrived at. -
track mode: the patient has arrived to a new non-temporary location. Use the optional field
bed
to set the specific bed the patient has arrived at.
See the following examples:
Example of transit mode:
- track_departure:
mode: transit
destination_loc: Ward 1
- track_arrival:
mode: transit
loc: Ward 1
Example of temporary mode:
- track_departure:
mode: temporary
destination_loc: Hallway
- track_departure:
mode: temporary
destination_loc: X-RAY
- track_arrival:
mode: temporary
loc: X-RAY
is_temporary: true
- track_departure:
mode: temporary
destination_loc: O/R
- track_arrival:
mode: temporary
loc: Ward 1
Example of track mode:
- track_departure:
mode: track
destination_loc: Ward 2
- track_arrival:
mode: track
loc: Ward 1
An autogenerate
event inserts one or more Result steps into the pathway within
a specified time-frame and at a specific interval. Neither the position of, nor
Delays specified for this step have an effect on when the Result events are
generated.
This step is useful to easily generate multiple results for a patient, for instance, vital signs.
Example:
- autogenerate:
result:
order_profile: Vital Signs
from: -50h
to: 5h
every: 2h
-
from
andto
define the timeframe of the generated events. Both are inclusive and are absolute times starting from when the pathway start. Iffrom
is negative, results will be inserted into thehistorical_data
sectio: e.g., in the above example, the first result message will be generated when time is -50h and thus belongs to thehistorical_data
section. Ifto
andfrom
are equal, a single result will be inserted. -
every
is the interval at which results will be inserted. Ifevery
is larger than the interval betweenfrom
andto
, only one result will be inserted.The following autogenerate step generates two results, one at time 0h and one at time 10h:
- autogenerate: from: 0h to: 10h every: 10h
The following autogenerate step generates a single result at time 0h:
- autogenerate: from: 0h to: 10h every: 11h
-
result
is the Result step to be inserted into the pathway. Any parameters (e.g.,order_profile
) will be copied to the generated steps and will produce the same behaviour they would have in a regularresult
step.
A clinical_note
event sends a HL7 message containing a document with
information about a patient, which clinicians refer to as a clinical note.
Clinical notes include discharge notes, images, etc., and can have multiple
formats.
The resulting HL7 message is an ORU^R01
message and thus is similar to the
message generated by a Results event. The message contains a single result, with
the content of the document in the OBX-5-ObservationValue
field. The content
will be encoded in base64 if the document is pdf, jpg or png.
A clinical_note
requires the content_type
field to be set and it must be one
of txt
, rtf
, pdf
, jpg
and png
. If content_type
is txt
, the field
document_content
can contain the fixed value for the note.
The following event will generate an ORU^R01
message with an embedded pdf
document. The content will be one of the predefined pdf documents from the
directory linked by the argument
--sample_notes_directory
).
- clinical_note:
content_type: pdf
The following event will generate an ORU^R01
message with a document of type
Discharge notes and fixed text:
- clinical_note:
content_type: txt
document_content: "My super secret notes"
document_type: "Discharge notes"
If an document_id
is specified, an addendum can be sent on top of the existing note.
The following sends a clinical note with a pdf.
- clinical_note:
document_id: my-id
content_type: pdf
document_type: "Discharge notes"
document_title: "Patient Overview"
Using the same document_id = my-id
to send another clinical note will append a new
document to the existing document. Specifying a new document_title
and
document_type
in the new pathway will override the previous values. The
following adds a new OBX to the previous message with the new document content
and updates the document_title
and document_type
.
- clinical_note:
document_id: my-id
content_type: txt
document_type: "Updated Discharge notes"
document_title: "Updated Patient Overview"
A hardcoded_message
event sends a pre-loaded message from the folder
configured by the
hardcoded_messages_dir
argument. This
step requires a regex
field with a comma separated list of regular expressions
that match one or more message names from the pre-loaded set.
When Simulated Hospital runs a hardcoded_message
step, it matches the names of
the pre-loaded messages with the regular expression. If the regular expression
matches no messages, the step results in an error. If the regular expression
matches one or more messages, Simulated Hospital picks and sends a random
message from the matching set.
For instance, the following step sends a random pre-loaded hardcoded message that starts with the string "Invalid":
- hardcoded_message:
regex: Invalid.*,
Use the special regular expression .*
to match any pre-loaded hardcoded
message.
Pre-loaded hardcoded messages must have a specific format, see Data configuration. When Simulated Hospital sends a message, it sets the following values:
- "MSH.7 - Date/Time Of Message": The current timestamp.
- "MSH-10 Message Control ID": A unique identifier for the message.
- PID segment (if the message contains the "PID_SEGMENT_PLACEHOLDER" keyword): The PID segment of the current patient.
A generic
event allows developers to specify situations that aren't covered by
other events, such as experimental code or events not supported yet in Simulated
Hospital.
Generic events do not have a default behavior. Developers need to pair all generic events with matching Override Event Processors. Generic events without matching processors will fail to run.
Use the parameters.custom
field to pass custom parameters to the events. Use
the name
field to distinguish between different types of generic events.
The following example uses two generic events. The interpretation and the behavior in each event is up to the developer to define, and is sent with event processors (see example). For instance, the first generic event could add a medication to the patient's record, and the second one a diagnosis.
- generic:
name: add_medication
parameters:
custom:
medication_name: paracetamol
- generic:
name: add_diagnosis
parameters:
custom:
diagnosis_name: appendicitis
A generate_resources
event will trigger the generation of the entire patient
record (at the time of the event) for each pathway as FHIR resources.
Currently, the following resources are supported:
Bundle
type
entry
Patient
identifier
name
gender
telecom
deceased
address
AllergyIntolerance
type
category
reaction
code
patient
clinicalStatus
Encounter
status
statusHistory
period
diagnoses
class
Observation
code
encounter
status
value
note
subject
Location
name
Procedure
code
category
subject
encounter
performed
performer
statusCode
Condition
code
subject
encounter
recordedDate
recorder
Order profiles define the type of results that are generated. All order profiles are loaded from a configuration file.
The yaml configuration file has the following format:
UREA AND ELECTROLYTES:
test_types:
Creatinine:
id: lpdc-2012
ref_range: 49 - 92
unit: UMOLL
value: '382'
value_type: NM
Potassium:
id: lpdc-2804
ref_range: 3.5 - 5.1
unit: MMOLL
value: '3.6'
value_type: NM
universal_service_id: lpdc-3969
Vital Signs:
test_types:
AVPU:
id: tt-0005-01
value: Alert
value_type: TX
BowelMovement:
id: tt-0005-02
value: 'Yes'
value_type: TX
[etc.]
In reality, UREA AND ELECTROLYTES order profile consists of more than 5 test types, but for simplicity let's it contains Creatinine and Potassium only.
When specifying Order or Result step in the pathway, the Order Profile name needs to be specified. It is used to look up the Order Profile from the configuration file.
For the Order step, the matching order profile is only used to populate the correct universal service name and id. For the Result step, it is also used to populate data like reference ranges, or to generate the normal / abnormal value for the result. There are multiple ways of how Results may be specified.
It is recommended to be explicit when defining the Result in the pathway, ie: to include results for each test type in the pathway. Example:
result:
order_profile: UREA AND ELECTROLYTES
results:
- test_name: Creatinine
value: 150
unit: UML
reference_range: 145 - 550
- test_name: Potassium
value: 4.6
unit: MMOLL
reference_range: 3.5 - 5.1
This will cause the message with 2 results (2 OBX segments) to be generated, with exactly the same values as specified in the pathway. Note:
reference_range
is optional. If not specified, the reference range for test type from the order profile will be used.value
may either be numerical or textual value; if the value is numerical, eg: 70, 5.5, <0.25, >=3.5 etc., then the Unit must also be set; if the value is textual, the Unit cannot be set.
If the scenario is focused on one particular test types, e.g. Creatinine for scenarios that measure Acute Kidney Injury, some test types may be omitted from the pathway. Note that Results for test types not specified in the pathway will not be included.
Example:
result:
order_profile: UREA AND ELECTROLYTES
results:
- test_name: Creatinine
value: 700
unit: UML
reference_range: 145 - 550
This will cause the message to be generated with one Creatinine result only, and no Potassium result. This may lead to less realistic pathways, as normally all test types for the order profile are included.
Results may be generated randomly based on the reference range. There are 3 options to generate random values: normal, abnormally high or abnormally low:
result:
order_profile: UREA AND ELECTROLYTES
results:
- test_name: Creatinine
value: 150
unit: UML
reference_range: 145 - 550
- test_name: Potassium
value: NORMAL
result:
order_profile: UREA AND ELECTROLYTES
results:
- test_name: Creatinine
value: 150
unit: UML
reference_range: 145 - 550
- test_name: Potassium
value: ABNORMAL_HIGH
result:
order_profile: UREA AND ELECTROLYTES
results:
- test_name: Creatinine
value: 150
unit: UML
reference_range: 145 - 550
- test_name: Potassium
value: ABNORMAL_LOW
This will cause the message to be generated with 2 results:
- Creatinine result with the value specified in the pathway
- Potassium result with the value randomly generated from the normal, high or low ranges (respectively) for this test type
If the value is specified as NORMAL, ABNORMAL_HIGH or ABNORMAL_LOW, then:
-
unit
must be omitted or also set to NORMAL, ABNORMAL_HIGH or ABNORMAL_LOW. However, if the reference range is specified, then the unit must also be specified. -
reference range
can be specified and a random value will be generated based on it. This reference range overrides the reference range from the order profile for the current message
If no results are specified, the results for each test type will be generated with a random value from the normal range of the order profile.
Example:
result:
order_profile: UREA AND ELECTROLYTES
This will cause the message to be generated with the full set of results from the Urea and Electrolytes profile: Creatinine and Potassium in this case. All of them will have a random value from their normal ranges.
The order profile may be specified as RANDOM, in which case a set of random (normal) results for the random order profile will be generated.
Example:
result:
order_profile: RANDOM
If there is no matching order profile found for the name specified in the pathway, the results will still be generated as far as the value and unit are set explicitly. For random values, reference ranges and units are required.
If there is a matching order profile for the name specified in the pathway, but the test names do not match the test names in the order profile, the pathway will be considered invalid.
When generating random values for Test Types for Order Profiles, there is a set of rules that are used:
- If the value type is not numerical, the value is always set to exactly the same textual value as in the order profile config file
- If the value is numerical and the reference ranges can be parsed, the value is randomly generated from the normal range
- If the value is numerical but the reference range cannot be parsed, the value is always set to the value from the order profile file
Numerical values are of the form: 70, 5.5, >0.25, <=5.5.
Reference ranges is an interval, within which the result is being considered normal. It can be defined in one of the following format:
from-to
from - to
[ from - to ]
[from-to]
from-to^from^to
where both: from
and to
are either positive or negative floating point
numbers;
or:
<to^^<to
<to^<to
<=to^^<=to
<=to^<=to
[ < to ]
[ <= to ]
[<to]
[<=to]
where to
is either positive or negative floating point number; from
is not
specified, meaning that the start of the range is open;
or:
>from^^>from
>from^>from
>=from^^>=from
>=from^>=from
[ > from ]
[ >= from ]
[>from]
[>=from]
where from
is either positive or negative floating point number; to
is not
specified, meaning that the end of the range is open.
When generating a normal random value from the reference range, the value is always strictly within the range. Similar when generating the abnormal high or abnormal low values - they are always strictly above or below the reference range respectively.
When generating a normal value from half-open reference range, we always make sure the generated value has a correct sign and is of a reasonable order of magnitute. Eg:
If the reference range is right open, ie: >from
, the normal value will be
generated using the following rules:
- if
from
is positive, the normal value is generated from (from
,10 x from
). - if
from
is negative, the normal value is generated from (from
,0
) to only allow negative numbers. - if
from
is0
, the normal value is generated from (from
,10
); this is an arbitrary choice, as we don't really know what order of magnitude the values should be in.
If the reference range is left open, ie: <to
, the normal value will be
generated using the following rules:
- if
to
is positive, the normal value is generated from (0
,to
) to only allow positive values. - if
to
is negative, the normal value is generated from (10 x to
,to
). - if
to
is 0, the normal value is generated from (-10
,to
); this is an arbitrary choice, as we don't really know what order of magnitude the values should be in.
Similar rules apply when generating abnormal high / abnormal low values from any reference range. We always make sure the generated number is of the reasonable order of magnitite and has the same sign as the high / low edge or the range, respecivelly.
It is not possile to generate abnormal high value for right open reference range, or abnormal low value for left open reference range. Attempting to do so will result in an error.
If the pathway contains multiple orders and results, each order and result must
have the order_id
set, to pair them together. The Result will use the same
data (placer number, order date, etc.) as the corresponding Order.
If the Result is specified but no order with the same order_id
was specified
in the pathway, then the simulator will assume that it belongs to a different
order and thus all Order specific data (placer number, order date, etc.) will be
generated when creating the result.
The first result with the order_id
is by default treated as final result. All
consequent results are then treated as corrections. The result status can
alternatively be overridden in the pathway.
The following steps generate two messages that relate to the same order, each message with one result. The result in the second message will be marked as a correction of the result in the first message:
- result:
order_id: 123
order_profile: CRP
results:
- test_name: Serum C-Reactive Protein
value: 40
unit: MGL
abnormal_flag: HIGH
- result:
order_id: 123
order_profile: CRP
results:
- test_name: Serum C-Reactive Protein
value: 44
unit: MGL
abnormal_flag: HIGH
Every new message with results for the same order has incremental numbers in the OBX.SetID fields. In the example above, the OBX.1 - Set ID field for the two messages will be 1 and 2, respectively.
If you need two steps related to the same order to keep the same SetID (for
instance, some downstream systems might require the same SetID in order to
process corrections or amendments) set the expect_correction
field of a
result
. This field indicates that we expect a correction or amendment for the
same order.
The next message for the same order will be treated as the amendment, and the SetIDs in both the initial message and the amendment will start from the same number. Note that downstream systems might require the results in the amendment to be specified in the same order as the original message. Whis can be achieved by specifying the results explicitly in the same order in both events. If results are generated randomly, the order is not guaranteed.
If the downstream processing systems treat preliminary values for results in a
different way, use expect_correction
together with the appropriate value for
order_status
and result_status
. Set order_status
and result_status
in
the last message to set a value different from the default value for a
correction. Example:
- result:
order_id: 123
order_profile: CRP
expect_correction: true
order_status: P
results_status: P
results:
- test_name: Serum C-Reactive Protein
value: 40
unit: MGL
abnormal_flag: HIGH
- result:
order_id: 123
order_profile: CRP
order_status: F
results_status: F
results:
- test_name: Serum C-Reactive Protein
value: 44
unit: MGL
abnormal_flag: HIGH
The parameters of the resulting messages for this example are the following:
-
First message:
- OBR.25 (Result Status): P
- OBX.1 (Set ID) = 1
- OBX.11 (Observation Result Status) = P
-
Second message:
- OBR.25 (Result Status): F
- OBX.1 (Set ID) = 1
- OBX.11 (Observation Result Status) = F
Subsequent messages for the same order will start with Set ID 2.
Each step can contain a parameters
field with the following parameters:
delay_message
: the delay between when the event happened, and when the HL7 message should be sent. This can be used to simulate delays in the hospital systems or messages out of order.time_from_now
: the time offset between now and when the event happened. This is mostly used for historical data with a negative value.death_indicator
: indication of death status change: practice is to use “N” or “” to declare undead, and “Y” or “DECEASED” to declare dead. If death declared, time_since_death must be specified.time_since_death
: (positive) time offset from the moment of death until now.sending_application
: the sending application (MSH.3) to set in the message related to this event.sending_facility
: the sending facility (MSH.4) to set in the message related to this event.receiving_application
: the receiving application (MSH.5) to set in the message related to this event.receiving_facility
: the receiving facility (MSH.6) to set in the message related to this event.custom
: a map of strings to strings, which can be used to pass arbitrary values for custom processing, see Custom event and message processors.
Example:
sample_pathway:
pathway:
- admission:
loc: "Ward"
parameters:
delay_message:
from: 1s
to: 60m
sending_application: MyEHR
custom:
my_arbitrary_field: my_arbitrary_value
A list of allergies can be specified in the following steps: update_person
,
admission
, discharge
, registration
, pre_admission
, discharge_in_error
,
and add_person
. The Allergies can be also explicitly set to an empty list to
indicate that the patient does not have allergies. If the allergies
section is
not set, allergies are randomly generated based on the probability that a
patient will have allergies. This probability, together with the maximum
allegies to generate per patient, are specified in the data.yml
configuration
file.
Allergies are included as AL1 segments in the corresponding ADT message.
Example:
allergies:
- type: MEDICATION
code: J30.1
description: Allergic rhinitis due to pollen
severity: MODERATE
reaction: Skin rash
- type: FOOD
code: K52.2
description: Allergic and dietetic gastroenteritis and colitis
severity: SEVERE
reaction: Rash
Simulated Hospital has some default locations defined in locations.yml
. All
event types for which a location needs to be specified (e.g., an admission
event) need to refer to an existing location from that configuration file.
A pathway that refers to an unknown location fails validation. See configure data for more information.
The following table describes what pathway events generate each type of HL7v2 message:
Message Type | Segments | Pathway Event |
---|---|---|
ADT^A01 | MSH, EVN, PID, PD1, PV1, NK1, AL1 | admission |
ADT^A02 | MSH, EVN, PID, PD1, PV1 | transfer_in_error |
ADT^A03 | MSH, EVN, PID, PD1, PV1, AL1 | discharge, discharge_in_error |
ADT^A04 | MSH, EVN, PID, PD1, PV1, NK1, AL1 | registration |
ADT^A05 | MSH, EVN, PID, PD1, PV1, PV2, NK1, AL1, DG1 | pre_admission |
ADT^A08 | MSH, EVN, PID, PD1, PV1, AL1, DG1, PR1 | update_person |
ADT^A09 | MSH, EVN, PID, PD1, PV1 | track_departure |
ADT^A10 | MSH, EVN, PID, PD1, PV1 | track_arrival |
ADT^A11 | MSH, EVN, PID, PD1, PV1 | cancel_visit |
ADT^A12 | MSH, EVN, PID, PD1, PV1 | cancel_transfer |
ADT^A13 | MSH, EVN, PID, PD1, PV1 | cancel_discharge |
ADT^A14 | MSH, EVN, PID, PD1, PV1, PV2 | pending_admission |
ADT^A15 | MSH, EVN, PID, PD1, PV1 | pending_transfer |
ADT^A16 | MSH, EVN, PID, PD1, PV1, PV2 | pending_discharge |
ADT^A17 | MSH, EVN, PID, PD1, PV1, PID, PD1, PV1 | bed_swap |
ADT^A23 | MSH, EVN, PID, PV1 | delete_visit |
ADT^A25 | MSH, EVN, PID, PD1, PV1, PV2 | cancel_pending_discharge |
ADT^A26 | MSH, EVN, PID, PD1, PV1, PV2 | cancel_pending_transfer |
ADT^A27 | MSH, EVN, PID, PD1, PV1, PV2 | cancel_pending_admission |
ADT^A28 | MSH, EVN, PID, PD1, PV1, AL1 | add_person |
ADT^A31 | MSH, EVN, PID, PD1, PV1, AL1, DG1, PR1 | update_person |
ADT^A34 | MSH, EVN, PID, PD1, MRG | merge |
ADT^A40 | MSH, EVN, PID, PD1, MRG, PV1 | merge |
MDM^T02 | MSH, EVN, PID, PV1, TXA, OBX | document |
ORM^O01 | MSH, PID, PV1, ORC, OBR, NTE, OBX, NTE | order |
ORR^O02 | MSH, MSA, PID, ORC | order |
ORU^R01 | MSH, PID, PV1, ORC, OBR, OBX, NTE | results, clinical_note |
ORU^R03 | MSH, PID, PV1, ORC, OBR, OBX, NTE | results |
ORU^R32 | MSH, PID, PV1, ORC, OBR, OBX, NTE | results |