This is the codebase for a digital implementation of the "FĂĽSim MANV" (FĂĽhrungssimulation Massenanfall von Verletzen), a German simulation system for training emergency medical services leadership personnel on how to manage Mass Casualty Incidents.
You can try it out at https://fuesim-manv.de/.
A screenshot of a part of an MCI exercise with initially ca. 50 patients at the Brandenburg Gate.
The concept is as follows:
- A trainer creates an exercise, which consists of patients, vehicles, viewports, transferPoints and other objects placed on a map.
- Participants can then join the exercise.
- The trainer can restrict the participants to a specific viewport. The participant cannot move out of this area.
- Vehicles (containing material, personnel and (sometimes) patients) can be transferred to other areas via transferPoints.
- After the exercise is started, patients that are not adequately treated by personnel and material can deteriorate and die. The goal of the participants is to prevent the patients from dying and transport them to the hospitals. To do this effectively they have to communicate with each other (via real radio devices, or remote via third-party services) and make the right decisions.
- Afterward, the exercise can be evaluated via statistics and a "time-travel" feature.
This simulation has been designed in cooperation with and with support from the Federal Academy for Civil Protection and Civil Defence of the Federal Office of Civil Protection and Disaster Assistance Germany, who are the original copyright holders of the analog "FĂĽSim MANV" simulation system, and the Malteser Hilfsdienst e.V. Berlin.
The simulation is implemented as a web application with an Angular frontend and NodeJS backend.
This project is currently developed as a bachelor project at the HPI. You can find the official project website here.
- Make sure to have git lfs installed.
- Install NodeJs (at least version 16.x) (if you need different node versions on your machine we recommend nvm or nvm for windows)
- npm should already come with NodeJs - if not install it
- Clone this repository
- Run
npm run setup
from the root folder - Copy the
.env.example
file to./.env
and adjust the settings as you need them. Note that some of the variables are explained under the next point. - Choose whether you want to use a database: You can (optionally) use a database for the persistence of exercise data. Look at the relevant section in the backend README for further information. Note that to not use the database you have to edit an environment variable, see the relevant section.
- (Optional) We have a list of recommended vscode extensions. We strongly recommend you to use them if you are developing. You can see them via the
@recommended
filter in the extensions panel.
If you are using vscode, you can run the task Start all
to start everything in one go.
Note that this tries to start the database using docker compose
. In case this fails please start the database in another way (see this section in the backend README).
- Open a terminal in
/shared
and runnpm run watch
- Open another terminal in
/frontend
and runnpm run start
- Open another terminal in
/backend
and runnpm run start
- Consider the database -- see point 7 of the installation.
You need to have docker
installed.
docker compose
needs to be installed. Note that, depending on your setup, you may usedocker-compose
instead ofdocker compose
. In this case, just replace the space in the commands with a dash (-
). For more information, see the relevant section of the documentation.- Run
docker compose up -d
in the root directory. This also starts the database. If you don't want to start the database rundocker compose up -d digital-fuesim-manv
instead.
- Execute
docker run -p -d 80:80 digitalfuesimmanv/dfm
.
The server will start listening using nginx on port 80
for all services (frontend, API, WebSockets).
Note the database requirements depicted in the installation section.
- Uncomment the build section of the docker compose file.
- Run
docker compose build
- Run
docker build -f docker/Dockerfile -t digital-fuesim-manv .
- All important volumes are listed in the docker-compose file.
- All available Docker ENVs are listed with their default values in .env.example file. Copy this file and name it
.env
(under Linux, this would be e.g.cp .env.example .env
)
- We are using git lfs. You can see the file types that currently use git lfs in .gitattributes. If you add another binary (or very large) file type to the repository you should add it there too.
- To see the images stored in git lfs in diff views in vscode we recommend running the following command once:
git config diff.lfs.textconv cat
. - We are using prettier as our code formatter. Run it via
npm run prettier
ornpm run prettier:windows
in the root to format all files and make the CI happy. Please use the vscode extension. - We are using eslint as our linter. Run it via
npm run lint:fix
in the root to lint (and auto fix if possible) all files. Please use the vscode extension.
There are already the following debug configurations for vscode saved:
- [frontend] Launch Chrome against localhost
- Debug Jest Tests
In addition you can make use of the following browser extensions:
We are using Jest for our unit tests.
You can run it during development
- from the terminal via
npm run test:watch
in the root,/shared
,/backend
or/frontend
folder - or via the recommended vscode extension.
We are using cypress to run the end-to-end tests. You can find the code under /frontend/cypress
in the repository.
To run the tests locally, it is recommended to use the vscode task Start all & cypress
. Alternatively, you can start the frontend and backend manually and then run npm run cy:open
in /frontend
.
If you only want to check whether the tests pass, you can run npm run cy:run
in /frontend
instead.
We are also making use of visual regression tests via cypress-image-diff.
The screenshots are stored under /frontend/cypress-visual-screenshots
.
The baseline
folder contains the reference screenshots (the desired result).
If a test fails a new screenshot is taken and put in the comparison
folder.
If the new screenshot is the new desired result, then you only have to move it in the baseline
folder and replace the old reference screenshot with the same name.
In the diff
folder you can see the changes between the baseline and the comparison screenshot.
- names are never unique, ids are
- private properties that may be used with getters/setters (and only those!) start with one leading underscore (
_
) dependencies
should be used for packages that must be installed when running the app (e.g.express
), whereasdevDependencies
should be used for packages only required for developing, debugging, building, or testing (e.g.jest
), which includes all@types
packages. We try to follow this route even for the frontend and the backend, although it is not important there. See also this answer on StackOverflow for more information about the differences.- Use JSDoc features for further documentation because editors like VSCode can display them better.
- Be aware that JSDoc comments must always go above the Decorator of the class/component/function/variable etc.
/** * Here is a description of the class/function/variable/etc. * * @param myParam a description of the parameter * @returns a nice variable that is bigger than {@link myVariable} * @throws myError when something goes wrong */
- You should use the keyword
TODO
to mark things that need to be done later. Whether an issue should be created is an individual decision.
This repository is a monorepo that consists of the following packages:
- frontend the browser-based client application (Angular)
- backend the server-side application (NodeJs)
- shared the shared code that is used by both frontend and backend
Each package has its own README.md
file with additional documentation. Please check them out before you start working on the project.
One server can host multiple exercises. Multiple clients can join an exercise. A client can only join one exercise at a time.
This is a real-time application.
Each client is connected to the server via a WebSocket connection. This means you can send and listen for events over a two-way communication channel. Via socket.io it is also possible to make use of a more classic request-response API via acknowledgments.
We borrow these core concepts from Redux.
A JSON object is an object whose properties are only the primitives string
, number
, boolean
or null
or another JSON object or an array of any of these (only state - no functions
).
Any object reference can't occur more than once anywhere in a JSON object (including nested objects). This means especially that no circular references are possible.
An immutable object is an object whose state cannot be modified after it is created. In the code immutability is conveyed via typescripts readonly and the helper type Immutable<T>
.
A state is an immutable JSON object. Each client as well as the server has a global state for an exercise. The single point of truth for all states of an exercise is the server. All these states should be synchronized.
You can find the exercise state here.
An action is an immutable JSON object that describes what should change in a state. The changes described by each action are atomic - this means either all or none of the changes described by an action are applied.
Actions cannot be applied in parallel. The order of actions is important.
It is a bad practice to encode part of the state in the action (or values derived/calculated from it). Instead, you should only read the state in the accompanying reducer.
A reducer is a pure function (no side effects!) that takes a state and an action of a specific type and returns a new state where the changes described in the action are applied. A state can only be modified by a reducer.
To be able to apply certain optimizations, it is advisable (but not necessary or guaranteed) that the reducer only changes the references of properties that have been changed.
You can find all exercise actions and reducers here. Please orient yourself on the already implemented actions, and don't forget to register them in shared/src/store/action-reducers/action-reducers.ts
It isn't necessary to copy the whole immutable object by value if it should be updated. Instead, only the objects that were modified should be shallow copied recursively. Immer provides a simple way to do this.
Because the state is immutable and reducers (should) only update the properties that have changed, you can short circuit in comparisons between immutable objects, if the references of objects in a property are equal. Therefore it is very performant to compare two states in the same context.
To save a state it is enough to save its reference. Therefore it is very performant as well. If the state would have to be changed, a new reference is created as the state is immutable.
Large values (images, large text, binary, etc.) are not directly stored in the state. Instead, the store only contains UUIDs that identify the blob. The blob can be retrieved via a separate (yet to be implemented) REST API.
The blob that belongs to a UUID cannot be changed or deleted while the state is still saved on the server. To change a blob, a new one should be uploaded and the old UUID in the state replaced with the new one.
If an action would add a new blobId to the state, the blob should have previously been uploaded to the server.
A blob should only be downloaded on demand (lazy) and cached.
- A client gets a snapshot of the state from the server via
getState
. - Any time an action is applied on the server, it is sent to all clients via
performAction
and applied to them too. Due to the maintained packet ordering via a WebSocket and the fact that the synchronization of the state in the backend works synchronously, it is impossible for a client to receive actions out of order or receive actions already included in the state received bygetState
. - A client can propose an action to the server via
proposeAction
. - If the proposal was accepted, the action is applied on the server and sent to all clients via
performAction
. - The server responds to a proposal with a response that indicates a success or rejection via an acknowledgment. A successful response is always sent after the
performAction
was broadcasted.
A consequence of the synchronization strategy described before is that it takes one roundtrip from the client to the server and back to get the correct state on the client that initiated the action. This can lead to a bad user experience because of high latency.
This is where optimistic updates come into play. We just assume optimistically that the proposed action will be applied on the server. Therefore we can apply the action on the client directly without waiting for a performAction
from the server.
If the server rejects the proposal or a race condition occurs, the client corrects its state again. In our case the optimisticActionHandler encapsulates this functionality.
The state in the frontend is not guaranteed to be correct. It is only guaranteed to automatically correct itself.
If you need to read from the state to change it, you should do this inside the action reducer because the currentState
passed into a reducer is always guaranteed to be correct.
- Do not save a very large JS primitve (a large string like a base64 encoded image) in a part of the state that is often modified (like the root). This primitive would be copied on each change. Instead, the primitive should be saved as part of a separate object. This makes use of the performance benefits of shallow copies.
- Currently, every client maintains the whole state, and every action is sent to all clients. There is no way to only subscribe to a part of the state and only receive updates for that part.
- License information about used images can be found here. All images are licensed under their original license.