Compare commits
206 Commits
Author | SHA1 | Date | |
---|---|---|---|
22ebffc761 | |||
3a19212a78 | |||
e95fa2fb12 | |||
1bcc8c13de | |||
2fc3299543 | |||
5575867632 | |||
87658bfa58 | |||
99efd6c98a | |||
e42d11f125 | |||
8b194e7212 | |||
070100eba0 | |||
ce1489d9a1 | |||
ca1e73a0b5 | |||
041bf459ef | |||
c922e27758 | |||
4d5c0ce1fb | |||
bb0bbf5a28 | |||
1b4585d601 | |||
dad3ea80ee | |||
01446bdece | |||
43660a4dcb | |||
8313420de5 | |||
1651f446fe | |||
056a087065 | |||
a730ce2a53 | |||
12eb1feea6 | |||
af714f7b64 | |||
f47b0b7de4 | |||
d8ba6b67cb | |||
40b89dd4f1 | |||
e3aeee61de | |||
07e59f9b02 | |||
f751d64220 | |||
299bd56d63 | |||
a0c4d1863f | |||
fc1901211d | |||
bed91439db | |||
a8e008000e | |||
a9c771aa99 | |||
256d9d4ff0 | |||
41b87835ca | |||
3654962294 | |||
9382da7101 | |||
67ab0ef2be | |||
69d796dab4 | |||
4bfe7dc698 | |||
0aec5e4479 | |||
b1a83510a6 | |||
6806c4fd69 | |||
b2016df852 | |||
b332f3c5ff | |||
9eb65788e1 | |||
9392bcadc9 | |||
a008641192 | |||
5928d52237 | |||
1eb40b40c9 | |||
feb9f2430c | |||
200c5338ea | |||
5c15ad4518 | |||
0b6decf88d | |||
cfdf41d550 | |||
45c65d53ce | |||
4568f491f5 | |||
952ad8c0cf | |||
3e45bf8494 | |||
46ddaf10f7 | |||
408b121f26 | |||
61e5273158 | |||
50b8f7bbd4 | |||
25f286f73b | |||
50a6d66718 | |||
f5c80f6d81 | |||
7659939d3c | |||
f58d4a76cf | |||
c2dd450579 | |||
a1b62ad754 | |||
59d8744c84 | |||
d2a6ec17ea | |||
550403cc9f | |||
d3a4500568 | |||
2e4fee97a8 | |||
5355eee05c | |||
3e22000541 | |||
8c319197d0 | |||
a31d2b4bcc | |||
67d5fb4c67 | |||
329be65d1a | |||
91fe3d1c23 | |||
f4b86ce2ea | |||
19d0daa442 | |||
cc9811d11d | |||
8ce5b06823 | |||
3cc34fb30b | |||
17e04a3f89 | |||
f71a775e12 | |||
45c83e943b | |||
6dcbfde62e | |||
4cdc419478 | |||
90b529adb4 | |||
a3bc60986b | |||
f5df0b5d22 | |||
972ac745e9 | |||
358373cf70 | |||
27a62321fa | |||
30cf0fd22e | |||
531a8589db | |||
fa89a64ddd | |||
45ad5e8827 | |||
c4eb4d0fe2 | |||
4bc69a353c | |||
9d30f750f7 | |||
a1a252d5a9 | |||
568942bfe5 | |||
15f0432553 | |||
113bf2dd2e | |||
7ac151202a | |||
5d9d47c2df | |||
585468314c | |||
441bff3783 | |||
21959c1698 | |||
8a11e6e85b | |||
5579ad1453 | |||
c2026bdd07 | |||
de6faecb02 | |||
3be8bc53ff | |||
fad2f33fd6 | |||
d88e2973da | |||
af767e4ea6 | |||
f98c970348 | |||
75872a149f | |||
e24ba430a5 | |||
08914a6f86 | |||
104f50afcb | |||
0083e75940 | |||
c892ff2461 | |||
4a9cffbaa5 | |||
8a6f9a6e02 | |||
91f17f6af5 | |||
8d4497bf2c | |||
4ab20a5f16 | |||
167587a473 | |||
e5d80f89b0 | |||
5d0e815037 | |||
a5a19e0cea | |||
1493a63e02 | |||
fe927e65aa | |||
add09c3f9c | |||
5eb969c36a | |||
3cc4f8c1a4 | |||
707bc55ab6 | |||
d7949a7dce | |||
f5999ff325 | |||
a62da60809 | |||
ced6609d9a | |||
8dee349c37 | |||
3e45de56cf | |||
7f54efe034 | |||
effcdd811f | |||
acf49a892e | |||
284806d130 | |||
cf2d338e13 | |||
d5552b3ca4 | |||
892c0dea8f | |||
0305e69e9e | |||
1a913b2644 | |||
0eee1908df | |||
ffea9343c8 | |||
eb24995ed9 | |||
4196664060 | |||
2824951e5e | |||
1e1db1c4d9 | |||
7440fe1e23 | |||
3f5c5e28fa | |||
6397b2a019 | |||
bf8f87b261 | |||
2f32834de0 | |||
79709caa39 | |||
c52509054d | |||
8fd587c2a3 | |||
edafe30a4b | |||
e24be0d325 | |||
5e93e834ad | |||
5e5bd579fb | |||
a24f869c84 | |||
635985bfd1 | |||
25143745c4 | |||
532254593f | |||
01ff53ab23 | |||
9643c80cc5 | |||
aa40da4995 | |||
da26b5a2c8 | |||
bbea48322f | |||
480f165c7b | |||
3d2c73ff8f | |||
9921e1e684 | |||
5bd26b894c | |||
8dc82225a4 | |||
2eb5cc61b9 | |||
78b2287163 | |||
66dc96680d | |||
64b8636145 | |||
2e7ef25a49 | |||
7186a45f6c | |||
72295202ec | |||
bc48a7217e | |||
a075f73162 |
20
.github/workflows/deploy.yml
vendored
@ -1,4 +1,4 @@
|
|||||||
name: "Run build and deploy"
|
name: 'Run build and deploy'
|
||||||
|
|
||||||
on:
|
on:
|
||||||
release:
|
release:
|
||||||
@ -8,20 +8,20 @@ jobs:
|
|||||||
docker:
|
docker:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-java@v3
|
- uses: actions/setup-java@v4
|
||||||
with:
|
with:
|
||||||
java-version: '17'
|
java-version: '21'
|
||||||
distribution: 'temurin'
|
distribution: 'temurin'
|
||||||
|
|
||||||
- name: Setup Gradle
|
- name: Setup Gradle
|
||||||
uses: gradle/gradle-build-action@v2.4.2
|
uses: gradle/actions/setup-gradle@v3
|
||||||
|
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v2
|
uses: docker/setup-buildx-action@v3
|
||||||
|
|
||||||
- name: Login to Docker Hub
|
- name: Login to Docker Hub
|
||||||
uses: docker/login-action@v2
|
uses: docker/login-action@v3
|
||||||
with:
|
with:
|
||||||
registry: ghcr.io
|
registry: ghcr.io
|
||||||
username: ${{ github.actor }}
|
username: ${{ github.actor }}
|
||||||
@ -30,6 +30,6 @@ jobs:
|
|||||||
- name: Execute image build and push
|
- name: Execute image build and push
|
||||||
run: |
|
run: |
|
||||||
./gradlew bootBuildImage
|
./gradlew bootBuildImage
|
||||||
docker tag ghcr.io/ccc-mf/etl-processor ghcr.io/ccc-mf/etl-processor:${{ github.ref_name }}
|
docker tag ghcr.io/${{ github.repository }} ghcr.io/${{ github.repository }}:${{ github.ref_name }}
|
||||||
docker push ghcr.io/ccc-mf/etl-processor
|
docker push ghcr.io/${{ github.repository }}
|
||||||
docker push ghcr.io/ccc-mf/etl-processor:${{ github.ref_name }}
|
docker push ghcr.io/${{ github.repository }}:${{ github.ref_name }}
|
16
.github/workflows/test.yml
vendored
@ -11,14 +11,14 @@ jobs:
|
|||||||
tests:
|
tests:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-java@v3
|
- uses: actions/setup-java@v4
|
||||||
with:
|
with:
|
||||||
java-version: '17'
|
java-version: '21'
|
||||||
distribution: 'temurin'
|
distribution: 'temurin'
|
||||||
|
|
||||||
- name: Setup Gradle
|
- name: Setup Gradle
|
||||||
uses: gradle/gradle-build-action@v2.4.2
|
uses: gradle/actions/setup-gradle@v3
|
||||||
|
|
||||||
- name: Execute tests
|
- name: Execute tests
|
||||||
run: ./gradlew test
|
run: ./gradlew test
|
||||||
@ -26,14 +26,14 @@ jobs:
|
|||||||
integrationTests:
|
integrationTests:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-java@v3
|
- uses: actions/setup-java@v4
|
||||||
with:
|
with:
|
||||||
java-version: '17'
|
java-version: '21'
|
||||||
distribution: 'temurin'
|
distribution: 'temurin'
|
||||||
|
|
||||||
- name: Setup Gradle
|
- name: Setup Gradle
|
||||||
uses: gradle/gradle-build-action@v2.4.2
|
uses: gradle/actions/setup-gradle@v3
|
||||||
|
|
||||||
- name: Execute integration tests
|
- name: Execute integration tests
|
||||||
run: ./gradlew integrationTest
|
run: ./gradlew integrationTest
|
3
.gitignore
vendored
@ -5,6 +5,8 @@ build/
|
|||||||
!**/src/main/**/build/
|
!**/src/main/**/build/
|
||||||
!**/src/test/**/build/
|
!**/src/test/**/build/
|
||||||
|
|
||||||
|
bindings/ca-certificates/*.pem
|
||||||
|
|
||||||
### STS ###
|
### STS ###
|
||||||
.apt_generated
|
.apt_generated
|
||||||
.classpath
|
.classpath
|
||||||
@ -36,3 +38,4 @@ out/
|
|||||||
### VS Code ###
|
### VS Code ###
|
||||||
.vscode/
|
.vscode/
|
||||||
/dev/gpas*
|
/dev/gpas*
|
||||||
|
/deploy/.env
|
||||||
|
332
README.md
@ -2,49 +2,196 @@
|
|||||||
|
|
||||||
Diese Anwendung versendet ein bwHC-MTB-File an das bwHC-Backend und pseudonymisiert die Patienten-ID.
|
Diese Anwendung versendet ein bwHC-MTB-File an das bwHC-Backend und pseudonymisiert die Patienten-ID.
|
||||||
|
|
||||||
## Pseudonymisierung der Patienten-ID
|
## Einordnung innerhalb einer DNPM-ETL-Strecke
|
||||||
|
|
||||||
Wenn eine URI zu einer gPAS-Instanz angegeben ist, wird diese verwendet.
|
Diese Anwendung erlaubt das Entgegennehmen von HTTP/REST-Anfragen aus dem Onkostar-Plugin **[onkostar-plugin-dnpmexport](https://github.com/CCC-MF/onkostar-plugin-dnpmexport)**.
|
||||||
|
|
||||||
|
Der Inhalt einer Anfrage, wenn ein bwHC-MTBFile, wird pseudonymisiert und auf Duplikate geprüft.
|
||||||
|
Duplikate werden verworfen, Änderungen werden weitergeleitet.
|
||||||
|
|
||||||
|
Löschanfragen werden immer als Löschanfrage an das bwHC-backend weitergeleitet.
|
||||||
|
|
||||||
|
Zudem ist eine minimalistische Weboberfläche integriert, die einen Einblick in den aktuellen Zustand der Anwendung gewährt.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
### Duplikaterkennung
|
||||||
|
|
||||||
|
Die Erkennung von Duplikaten ist normalerweise immer aktiv, kann jedoch über den Konfigurationsparameter
|
||||||
|
`APP_DUPLICATION_DETECTION=false` deaktiviert werden.
|
||||||
|
|
||||||
|
### Datenübermittlung über HTTP/REST
|
||||||
|
|
||||||
|
Anfragen werden, wenn nicht als Duplikat behandelt, nach der Pseudonymisierung direkt an das bwHC-Backend gesendet.
|
||||||
|
|
||||||
|
### Datenübermittlung mit Apache Kafka
|
||||||
|
|
||||||
|
Anfragen werden, wenn nicht als Duplikat behandelt, nach der Pseudonymisierung an Apache Kafka übergeben.
|
||||||
|
Eine Antwort wird dabei ebenfalls mithilfe von Apache Kafka übermittelt und nach der Entgegennahme verarbeitet.
|
||||||
|
|
||||||
|
Siehe hierzu auch: https://github.com/CCC-MF/kafka-to-bwhc
|
||||||
|
|
||||||
|
## Konfiguration
|
||||||
|
|
||||||
|
### Pseudonymisierung der Patienten-ID
|
||||||
|
|
||||||
|
Wenn eine URI zu einer gPAS-Instanz (Version >= 2023.1.0) angegeben ist, wird diese verwendet.
|
||||||
Ist diese nicht gesetzt. wird intern eine Anonymisierung der Patienten-ID vorgenommen.
|
Ist diese nicht gesetzt. wird intern eine Anonymisierung der Patienten-ID vorgenommen.
|
||||||
|
|
||||||
* `APP_PSEUDONYMIZE_PREFIX`: Standortbezogenes Prefix - `UNKNOWN`, wenn nicht gesetzt
|
* `APP_PSEUDONYMIZE_PREFIX`: Standortbezogenes Präfix - `UNKNOWN`, wenn nicht gesetzt
|
||||||
* `APP_PSEUDONYMIZE_GENERATOR`: `BUILDIN` oder `GPAS` - `BUILDIN`, wenn nicht gesetzt
|
* `APP_PSEUDONYMIZE_GENERATOR`: `BUILDIN` oder `GPAS` - `BUILDIN`, wenn nicht gesetzt
|
||||||
|
|
||||||
### Eingebaute Pseudonymisierung
|
**Hinweise**:
|
||||||
|
|
||||||
Wurde keine oder die Verwendung der eingebauten Pseudonymisierung konfiguriert, so wird für die Patienten-ID der
|
* Der alte Konfigurationsparameter `APP_PSEUDONYMIZER` mit den Werten `GPAS` oder `BUILDIN` sollte nicht mehr verwendet
|
||||||
entsprechende SHA-256-Hash gebildet und Base64-codiert - hier ohne endende "=" - zuzüglich des konfigurierten Prefixes
|
werden.
|
||||||
|
* Die Pseudonymisierung erfolgt im ETL-Prozessor nur für die Patienten-ID.
|
||||||
|
Andere IDs werden mithilfe des standortbezogenen Präfixes (erneut) anonymisiert, um für den aktuellen Kontext nicht
|
||||||
|
vergleichbare IDs bereitzustellen.
|
||||||
|
|
||||||
|
#### Eingebaute Anonymisierung
|
||||||
|
|
||||||
|
Wurde keine oder die Verwendung der eingebauten Anonymisierung konfiguriert, so wird für die Patienten-ID der
|
||||||
|
entsprechende SHA-256-Hash gebildet und Base64-codiert - hier ohne endende "=" - zuzüglich des konfigurierten Präfixes
|
||||||
als Patienten-Pseudonym verwendet.
|
als Patienten-Pseudonym verwendet.
|
||||||
|
|
||||||
### Pseudonymisierung mit gPAS
|
#### Pseudonymisierung mit gPAS
|
||||||
|
|
||||||
Wurde die Verwendung von gPAS konfiguriert, so sind weitere Angaben zu konfigurieren.
|
Wurde die Verwendung von gPAS konfiguriert, so sind weitere Angaben zu konfigurieren.
|
||||||
|
|
||||||
* `APP_PSEUDONYMIZE_GPAS_URI`: URI der gPAS-Instanz inklusive Endpoint (z.B. `http://localhost:8080/ttp-fhir/fhir/gpas/$pseudonymizeAllowCreate`)
|
* `APP_PSEUDONYMIZE_GPAS_URI`: URI der gPAS-Instanz inklusive Endpoint (z.B. `http://localhost:8080/ttp-fhir/fhir/gpas/$$pseudonymizeAllowCreate`)
|
||||||
* `APP_PSEUDONYMIZE_GPAS_TARGET`: gPas Domänenname
|
* `APP_PSEUDONYMIZE_GPAS_TARGET`: gPas Domänenname
|
||||||
* `APP_PSEUDONYMIZE_GPAS_USERNAME`: gPas Basic-Auth Benutzername
|
* `APP_PSEUDONYMIZE_GPAS_USERNAME`: gPas Basic-Auth Benutzername
|
||||||
* `APP_PSEUDONYMIZE_GPAS_PASSWORD`: gPas Basic-Auth Passwort
|
* `APP_PSEUDONYMIZE_GPAS_PASSWORD`: gPas Basic-Auth Passwort
|
||||||
* `APP_PSEUDONYMIZE_GPAS_SSLCALOCATION`: Root Zertifikat für gPas, falls es dediziert hinzugefügt werden muss.
|
* ~~`APP_PSEUDONYMIZE_GPAS_SSLCALOCATION`~~: **Veraltet** - Root Zertifikat für gPas, falls es dediziert hinzugefügt werden muss.
|
||||||
|
|
||||||
## Mögliche Endpunkte
|
Der Konfigurationsparameter `APP_PSEUDONYMIZE_GPAS_SSLCALOCATION` sollte nicht mehr verwendet werden und wird in einer kommenden Version entfernt.
|
||||||
|
Stattdessen sollte das Root Zertifikat wie unter [_Integration eines eigenen Root CA Zertifikats_](#integration-eines-eigenen-root-ca-zertifikats) beschrieben eingebunden werden.
|
||||||
|
|
||||||
|
### Anmeldung mit einem Passwort
|
||||||
|
|
||||||
|
Ein initialer Administrator-Account kann optional konfiguriert werden und sorgt dafür, dass bestimmte Bereiche nur nach
|
||||||
|
einem erfolgreichen Login erreichbar sind.
|
||||||
|
|
||||||
|
* `APP_SECURITY_ADMIN_USER`: Muss angegeben werden zur Aktivierung der Zugriffsbeschränkung.
|
||||||
|
* `APP_SECURITY_ADMIN_PASSWORD`: Das Passwort für den Administrator (Empfohlen).
|
||||||
|
|
||||||
|
Ein Administrator-Passwort muss inklusive des Encoding-Präfixes vorliegen.
|
||||||
|
|
||||||
|
Hier Beispiele für das Beispielpasswort `very-secret`:
|
||||||
|
|
||||||
|
* `{noop}very-secret` (Das Passwort liegt im Klartext vor - nicht empfohlen!)
|
||||||
|
* `{bcrypt}$2y$05$CCkfsMr/wbTleMyjVIK8g.Aa3RCvrvoLXVAsL.f6KeouS88vXD9b6`
|
||||||
|
* `{sha256}9a34717f0646b5e9cfcba70055de62edb026ff4f68671ba3db96aa29297d2df5f1a037d58c745657`
|
||||||
|
|
||||||
|
Wird kein Administrator-Passwort angegeben, wird ein zufälliger Wert generiert und beim Start der Anwendung in den Logs
|
||||||
|
angezeigt.
|
||||||
|
|
||||||
|
#### Weitere (nicht administrative) Nutzer mit OpenID Connect
|
||||||
|
|
||||||
|
Die folgenden Konfigurationsparameter werden benötigt, um die Authentifizierung weiterer Benutzer an einen OIDC-Provider
|
||||||
|
zu delegieren.
|
||||||
|
Ein Admin-Benutzer muss dabei konfiguriert sein.
|
||||||
|
|
||||||
|
* `APP_SECURITY_ENABLE_OIDC`: Aktiviert die Nutzung von OpenID Connect. Damit sind weitere Parameter erforderlich
|
||||||
|
* `SPRING_SECURITY_OAUTH2_CLIENT_REGISTRATION_CUSTOM_CLIENT_NAME`: Name. Wird beim zusätzlichen Loginbutton angezeigt.
|
||||||
|
* `SPRING_SECURITY_OAUTH2_CLIENT_REGISTRATION_CUSTOM_CLIENT_ID`: Client-ID
|
||||||
|
* `SPRING_SECURITY_OAUTH2_CLIENT_REGISTRATION_CUSTOM_CLIENT_SECRET`: Client-Secret
|
||||||
|
* `SPRING_SECURITY_OAUTH2_CLIENT_REGISTRATION_CUSTOM_CLIENT_SCOPE[0]`: Hier sollte immer `openid` angegeben werden.
|
||||||
|
* `SPRING_SECURITY_OAUTH2_CLIENT_PROVIDER_CUSTOM_ISSUER_URI`: Die URI des Providers,
|
||||||
|
z.B. `https://auth.example.com/realm/example`
|
||||||
|
* `SPRING_SECURITY_OAUTH2_CLIENT_PROVIDER_CUSTOM_USER_NAME_ATTRIBUTE`: Name des Attributes, welches den Benutzernamen
|
||||||
|
enthält.
|
||||||
|
Oft verwendet: `preferred_username`
|
||||||
|
|
||||||
|
Ist die Nutzung von OpenID Connect konfiguriert, erscheint ein zusätzlicher Login-Button zur Nutzung mit OpenID Connect
|
||||||
|
und dem konfigurierten `CLIENT_NAME`.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Weitere Informationen zur Konfiguration des OIDC-Providers
|
||||||
|
sind [hier](https://docs.spring.io/spring-security/reference/servlet/oauth2/index.html#oauth2-client)
|
||||||
|
zu finden.
|
||||||
|
|
||||||
|
#### Rollenbasierte Berechtigungen
|
||||||
|
|
||||||
|
Wird OpenID Connect verwendet, gibt es eine rollenbasierte Berechtigungszuweisung.
|
||||||
|
|
||||||
|
Die Standardrolle für neue OIDC-Benutzer kann mit der Option `APP_SECURITY_DEFAULT_USER_ROLE` festgelegt werden.
|
||||||
|
Mögliche Werte sind `user` oder `guest`. Standardwert ist `user`.
|
||||||
|
|
||||||
|
Benutzer mit der Rolle "Gast" sehen nur die Inhalte, die auch nicht angemeldete Benutzer sehen.
|
||||||
|
|
||||||
|
Hierdurch ist es möglich, einzelne Benutzer einzuschränken oder durch Änderung der Standardrolle auf `guest` nur
|
||||||
|
einzelne Benutzer als vollwertige Nutzer zuzulassen.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Benutzer werden nach dem Entfernen oder der Änderung der vergebenen Rolle automatisch abgemeldet und müssen sich neu anmelden.
|
||||||
|
Sie bekommen dabei wieder die Standardrolle zugewiesen.
|
||||||
|
|
||||||
|
#### Auswirkungen auf den dargestellten Inhalt
|
||||||
|
|
||||||
|
Nur Administratoren haben Zugriff auf den Konfigurationsbereich, nur angemeldete Benutzer können die anonymisierte oder
|
||||||
|
pseudonymisierte Patienten-ID sowie den Qualitätsbericht des bwHC-Backends einsehen.
|
||||||
|
|
||||||
|
Wurde kein Administrator-Account konfiguriert, sind diese Inhalte generell nicht verfügbar.
|
||||||
|
|
||||||
|
### Tokenbasierte Authentifizierung für MTBFile-Endpunkt
|
||||||
|
|
||||||
|
Die Anwendung unterstützt das Erstellen und Nutzen einer tokenbasierten Authentifizierung für den MTB-File-Endpunkt.
|
||||||
|
|
||||||
|
Dies kann mit der Umgebungsvariable `APP_SECURITY_ENABLE_TOKENS` aktiviert (`true` oder `false`) werden
|
||||||
|
und ist als Standardeinstellung nicht aktiv.
|
||||||
|
|
||||||
|
Ist diese Einstellung aktiviert worden, ist es Administratoren möglich, Zugriffstokens für Onkostar zu erstellen, die
|
||||||
|
zur Nutzung des MTB-File-Endpunkts eine HTTP-Basic-Authentifizierung voraussetzen.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
In diesem Fall können den Endpunkt für das Onkostar-Plugin **[onkostar-plugin-dnpmexport](https://github.com/CCC-MF/onkostar-plugin-dnpmexport)** wie folgt konfigurieren:
|
||||||
|
|
||||||
|
```
|
||||||
|
https://testonkostar:MTg1NTL...NGU4@etl.example.com/mtbfile
|
||||||
|
```
|
||||||
|
|
||||||
|
Ist die Verwendung von Tokens aktiv, werden Anfragen ohne die Angabe der Token-Information abgelehnt.
|
||||||
|
|
||||||
|
### Transformation von Werten
|
||||||
|
|
||||||
|
In Onkostar kann es vorkommen, dass ein Wert eines Merkmalskatalogs an einem Standort angepasst wurde und dadurch nicht dem Wert entspricht,
|
||||||
|
der vom bwHC-Backend akzeptiert wird.
|
||||||
|
|
||||||
|
Diese Anwendung bietet daher die Möglichkeit, eine Transformation vorzunehmen. Hierzu muss der "Pfad" innerhalb des JSON-MTB-Files angegeben werden und
|
||||||
|
welcher Wert wie ersetzt werden soll.
|
||||||
|
|
||||||
|
Hier ein Beispiel für die erste (Index 0 - weitere dann mit 1,2, ...) Transformationsregel:
|
||||||
|
|
||||||
|
* `APP_TRANSFORMATIONS_0_PATH`: Pfad zum Wert in der JSON-MTB-Datei. Beispiel: `diagnoses[*].icd10.version` für **alle** Diagnosen
|
||||||
|
* `APP_TRANSFORMATIONS_0_FROM`: Angabe des Werts, der ersetzt werden soll. Andere Werte bleiben dabei unverändert.
|
||||||
|
* `APP_TRANSFORMATIONS_0_TO`: Angabe des neuen Werts.
|
||||||
|
|
||||||
|
### Mögliche Endpunkte zur Datenübermittlung
|
||||||
|
|
||||||
Für REST-Requests als auch zur Nutzung von Kafka-Topics können Endpunkte konfiguriert werden.
|
Für REST-Requests als auch zur Nutzung von Kafka-Topics können Endpunkte konfiguriert werden.
|
||||||
|
|
||||||
Es ist dabei nur die Konfiguration eines Endpunkts zulässig.
|
Es ist dabei nur die Konfiguration eines Endpunkts zulässig.
|
||||||
Werden sowohl REST als auch Kafka-Endpunkt konfiguriert, wird nur der REST-Endpunkt verwendet.
|
Werden sowohl REST als auch Kafka-Endpunkt konfiguriert, wird nur der REST-Endpunkt verwendet.
|
||||||
|
|
||||||
### REST
|
#### REST
|
||||||
|
|
||||||
Folgende Umgebungsvariablen müssen gesetzt sein, damit ein bwHC-MTB-File an das bwHC-Backend gesendet wird:
|
Folgende Umgebungsvariablen müssen gesetzt sein, damit ein bwHC-MTB-File an das bwHC-Backend gesendet wird:
|
||||||
|
|
||||||
* `APP_REST_URI`: URI der zu benutzenden API der bwHC-Backend-Instanz. z.B.: `http://localhost:9000/bwhc/etl/api`
|
* `APP_REST_URI`: URI der zu benutzenden API der bwHC-Backend-Instanz. z.B.: `http://localhost:9000/bwhc/etl/api`
|
||||||
|
|
||||||
### Kafka-Topics
|
#### Kafka-Topics
|
||||||
|
|
||||||
Folgende Umgebungsvariablen müssen gesetzt sein, damit ein bwHC-MTB-File an ein Kafka-Topic übermittelt wird:
|
Folgende Umgebungsvariablen müssen gesetzt sein, damit ein bwHC-MTB-File an ein Kafka-Topic übermittelt wird:
|
||||||
|
|
||||||
* `APP_KAFKA_TOPIC`: Zu verwendendes Topic zum Versenden von Anfragen
|
* `APP_KAFKA_OUTPUT_TOPIC`: Zu verwendendes Topic zum Versenden von Anfragen.
|
||||||
* `APP_KAFKA_RESPONSE_TOPIC`: Topic mit Antworten über den Erfolg des Versendens. Standardwert: `APP_KAFKA_TOPIC` mit Anhang "_response".
|
Ersetzt in einer kommenden Version `APP_KAFKA_TOPIC`.
|
||||||
|
* `APP_KAFKA_OUTPUT_RESPONSE_TOPIC`: Topic mit Antworten über den Erfolg des Versendens. Standardwert: `APP_KAFKA_TOPIC` mit Anhang "_response".
|
||||||
|
Ersetzt in einer kommenden Version `APP_KAFKA_RESPONSE_TOPIC`.
|
||||||
* `APP_KAFKA_GROUP_ID`: Kafka GroupID des Consumers. Standardwert: `APP_KAFKA_TOPIC` mit Anhang "_group".
|
* `APP_KAFKA_GROUP_ID`: Kafka GroupID des Consumers. Standardwert: `APP_KAFKA_TOPIC` mit Anhang "_group".
|
||||||
* `APP_KAFKA_SERVERS`: Zu verwendende Kafka-Bootstrap-Server als kommagetrennte Liste
|
* `APP_KAFKA_SERVERS`: Zu verwendende Kafka-Bootstrap-Server als kommagetrennte Liste
|
||||||
|
|
||||||
@ -55,6 +202,161 @@ Weitere Einstellungen können über die Parameter von Spring Kafka konfiguriert
|
|||||||
Lässt sich keine Verbindung zu dem bwHC-Backend aufbauen, wird eine Rückantwort mit Status-Code `900` erwartet, welchen es
|
Lässt sich keine Verbindung zu dem bwHC-Backend aufbauen, wird eine Rückantwort mit Status-Code `900` erwartet, welchen es
|
||||||
für HTTP nicht gibt.
|
für HTTP nicht gibt.
|
||||||
|
|
||||||
|
Wird die Umgebungsvariable `APP_KAFKA_INPUT_TOPIC` gesetzt, kann eine Nachricht auch über dieses Kafka-Topic an den ETL-Prozessor übermittelt werden.
|
||||||
|
|
||||||
|
##### Retention Time
|
||||||
|
|
||||||
|
Generell werden in Apache Kafka alle Records entsprechend der Konfiguration vorgehalten.
|
||||||
|
So wird ohne spezielle Konfiguration ein Record für 7 Tage in Apache Kafka gespeichert.
|
||||||
|
Es sind innerhalb dieses Zeitraums auch alte Informationen weiterhin enthalten, wenn der Consent später abgelehnt wurde.
|
||||||
|
|
||||||
|
Durch eine entsprechende Konfiguration des Topics kann dies verhindert werden.
|
||||||
|
|
||||||
|
Beispiel - auszuführen innerhalb des Kafka-Containers: Löschen alter Records nach einem Tag
|
||||||
|
```
|
||||||
|
kafka-configs.sh --bootstrap-server localhost:9092 --alter --topic test --add-config retention.ms=86400000
|
||||||
|
```
|
||||||
|
|
||||||
|
##### Key based Retention
|
||||||
|
|
||||||
|
Möchten Sie hingegen immer nur die letzte Meldung für einen Patienten und eine Erkrankung in Apache Kafka vorhalten,
|
||||||
|
so ist die nachfolgend genannte Konfiguration der Kafka-Topics hilfreich.
|
||||||
|
|
||||||
|
|
||||||
|
* `retention.ms`: Möglichst kurze Zeit in der alte Records noch erhalten bleiben, z.B. 10 Sekunden 10000
|
||||||
|
* `cleanup.policy`: Löschen alter Records und Beibehalten des letzten Records zu einem Key [delete,compact]
|
||||||
|
|
||||||
|
Beispiele für ein Topic `test`, hier bitte an die verwendeten Topics anpassen.
|
||||||
|
|
||||||
|
```
|
||||||
|
kafka-configs.sh --bootstrap-server localhost:9092 --alter --topic test --add-config retention.ms=10000
|
||||||
|
kafka-configs.sh --bootstrap-server localhost:9092 --alter --topic test --add-config cleanup.policy=[delete,compact]
|
||||||
|
```
|
||||||
|
|
||||||
|
Da als Key eines Records die (pseudonymisierte) Patienten-ID verwendet wird, stehen mit obiger Konfiguration
|
||||||
|
der Kafka-Topics nach 10 Sekunden nur noch der jeweils letzte Eintrag für den entsprechenden Key zur Verfügung.
|
||||||
|
|
||||||
|
Da der Key sowohl für die Records in Richtung bwHC-Backend für die Rückantwort identisch aufgebaut ist, lassen sich so
|
||||||
|
auch im Falle eines Consent-Widerspruchs die enthaltenen Daten als auch die Offenlegung durch Verifikationsdaten in der
|
||||||
|
Antwort effektiv verhindern, da diese nach 10 Sekunden gelöscht werden.
|
||||||
|
|
||||||
|
Es steht dann nur noch die jeweils letzten Information zur Verfügung, dass für einen Patienten/eine Erkrankung
|
||||||
|
ein Consent-Widerspruch erfolgte.
|
||||||
|
|
||||||
|
Dieses Vorgehen empfiehlt sich, wenn Sie gespeicherte Records nachgelagert für andere Auswertungen verwenden möchten.
|
||||||
|
|
||||||
## Docker-Images
|
## Docker-Images
|
||||||
|
|
||||||
Diese Anwendung ist auch als Docker-Image verfügbar: https://github.com/CCC-MF/etl-processor/pkgs/container/etl-processor
|
Diese Anwendung ist auch als Docker-Image verfügbar: https://github.com/CCC-MF/etl-processor/pkgs/container/etl-processor
|
||||||
|
|
||||||
|
### Images lokal bauen
|
||||||
|
|
||||||
|
```bash
|
||||||
|
./gradlew bootBuildImage
|
||||||
|
```
|
||||||
|
|
||||||
|
### Integration eines eigenen Root CA Zertifikats
|
||||||
|
|
||||||
|
Wird eine eigene Root CA verwendet, die nicht offiziell signiert ist, wird es zu Problemen beim SSL-Handshake kommen, wenn z.B. gPAS zur Generierung von Pseudonymen verwendet wird.
|
||||||
|
|
||||||
|
Hier bietet es sich an, das Root CA Zertifikat in das Image zu integrieren.
|
||||||
|
|
||||||
|
#### Integration beim Bauen des Images
|
||||||
|
|
||||||
|
Hier muss die Zeile `"BP_EMBED_CERTS" to "true"` in der Datei `build.gradle.kts` verwendet werden und darf nicht als Kommentar verwendet werden.
|
||||||
|
|
||||||
|
Die PEM-Datei mit dem/den Root CA Zertifikat(en) muss dabei im vorbereiteten Verzeichnis [`bindings/ca-certificates`](bindings/ca-certificates) enthalten sein.
|
||||||
|
|
||||||
|
#### Integration zur Laufzeit
|
||||||
|
|
||||||
|
Hier muss die Umgebungsvariable `SERVICE_BINDING_ROOT` z.B. auf den Wert `/bindings` gesetzt sein.
|
||||||
|
Zudem muss ein Verzeichnis `bindings/ca-certificates` - analog zum Verzeichnis [`bindings/ca-certificates`](bindings/ca-certificates) mit einer PEM-Datei als Docker-Volume eingebunden werden.
|
||||||
|
|
||||||
|
Beispiel für Docker-Compose:
|
||||||
|
|
||||||
|
```
|
||||||
|
...
|
||||||
|
environment:
|
||||||
|
SERVICE_BINDING_ROOT: /bindings
|
||||||
|
...
|
||||||
|
volumes:
|
||||||
|
- "/path/to/bindings/ca-certificates/:/bindings/ca-certificates/:ro"
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Deployment
|
||||||
|
*Ausführen als Docker Container:*
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd ./deploy
|
||||||
|
cp env-sample.env .env
|
||||||
|
```
|
||||||
|
Wenn gewünscht, Änderungen in der `.env` vornehmen.
|
||||||
|
|
||||||
|
```bash
|
||||||
|
docker compose up -d
|
||||||
|
```
|
||||||
|
|
||||||
|
### Einfaches Beispiel für ein eigenes Docker-Compose-File
|
||||||
|
|
||||||
|
Die Datei [`docs/docker-compose.yml`](docs/docker-compose.yml) zeigt eine einfache Konfiguration für REST-Requests basierend
|
||||||
|
auf Docker-Compose mit der gestartet werden kann.
|
||||||
|
|
||||||
|
### Betrieb hinter einem Reverse-Proxy
|
||||||
|
|
||||||
|
Die Anwendung verarbeitet `X-Forwarded`-HTTP-Header und kann daher auch hinter einem Reverse-Proxy betrieben werden.
|
||||||
|
|
||||||
|
Dabei werden, je nachdem welche Header durch den Reverse-Proxy gesendet werden auch Protokoll, Host oder auch Path-Präfix
|
||||||
|
automatisch erkannt und verwendet werden. Dadurch ist z.B. eine abweichende Angabe des Pfads problemlos möglich.
|
||||||
|
|
||||||
|
#### Beispiel *Traefik* (mit Docker-Labels):
|
||||||
|
|
||||||
|
Das folgende Beispiel zeigt die Konfiguration in einer Docker-Compose-Datei mit Service-Labels.
|
||||||
|
|
||||||
|
```
|
||||||
|
...
|
||||||
|
deploy:
|
||||||
|
labels:
|
||||||
|
- "traefik.http.routers.etl.rule=PathPrefix(`/etl-processor`)"
|
||||||
|
- "traefik.http.routers.etl.middlewares=etl-path-strip"
|
||||||
|
- "traefik.http.middlewares.etl-path-strip.stripprefix.prefixes=/etl-processor"
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Beispiel *nginx*
|
||||||
|
|
||||||
|
Das folgende Beispiel zeigt die Konfiguration einer _location_ in einer nginx-Konfigurationsdatei.
|
||||||
|
|
||||||
|
```
|
||||||
|
...
|
||||||
|
location /etl-processor {
|
||||||
|
set $upstream http://<beispiel:8080>/;
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Forwarded-Scheme $scheme;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
proxy_set_header X-Forwarded-For $remote_addr;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_pass $upstream;
|
||||||
|
}
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
## Entwicklungssetup
|
||||||
|
|
||||||
|
Zum Starten einer lokalen Entwicklungs- und Testumgebung kann die beiliegende Datei `dev-compose.yml` verwendet werden.
|
||||||
|
Diese kann zur Nutzung der Datenbanken **MariaDB** als auch **PostgreSQL** angepasst werden.
|
||||||
|
|
||||||
|
Zur Nutzung von Apache Kafka muss dazu ein Eintrag im hosts-File vorgenommen werden und der Hostname `kafka` auf die lokale
|
||||||
|
IP-Adresse verweisen. Ohne diese Einstellung ist eine Nutzung von Apache Kafka außerhalb der Docker-Umgebung nicht möglich.
|
||||||
|
|
||||||
|
Beim Start der Anwendung mit dem Profil `dev` wird die in `dev-compose.yml` definierte Umgebung beim Start der
|
||||||
|
Anwendung mit gestartet:
|
||||||
|
|
||||||
|
```
|
||||||
|
SPRING_PROFILES_ACTIVE=dev ./gradlew bootRun
|
||||||
|
```
|
||||||
|
|
||||||
|
Die Datei `application-dev.yml` enthält hierzu die Konfiguration für das Profil `dev`.
|
||||||
|
|
||||||
|
Beim Ausführen der Integrationstests wird eine Testdatenbank in einem Docker-Container gestartet.
|
||||||
|
Siehe hier auch die Klasse `AbstractTestcontainerTest` unter `src/integrationTest`.
|
||||||
|
5
bindings/README.md
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
# Hinweis für Root CA Zertifikate
|
||||||
|
|
||||||
|
PEM-Datei(en) in das Verzeichnis `ca-certificates` ablegen.
|
||||||
|
|
||||||
|
Die Datei `type` gibt dabei an, dass hier CA Zertifikate zu finden sind.
|
1
bindings/ca-certificates/type
Normal file
@ -0,0 +1 @@
|
|||||||
|
ca-certificates
|
@ -4,17 +4,27 @@ import org.springframework.boot.gradle.tasks.bundling.BootBuildImage
|
|||||||
|
|
||||||
plugins {
|
plugins {
|
||||||
war
|
war
|
||||||
id("org.springframework.boot") version "3.1.2"
|
id("org.springframework.boot") version "3.2.12"
|
||||||
id("io.spring.dependency-management") version "1.1.0"
|
id("io.spring.dependency-management") version "1.1.5"
|
||||||
kotlin("jvm") version "1.9.0"
|
kotlin("jvm") version "1.9.24"
|
||||||
kotlin("plugin.spring") version "1.9.0"
|
kotlin("plugin.spring") version "1.9.24"
|
||||||
}
|
}
|
||||||
|
|
||||||
group = "de.ukw.ccc"
|
group = "dev.dnpm"
|
||||||
version = "0.1.1"
|
version = "0.9.8"
|
||||||
|
|
||||||
|
var versions = mapOf(
|
||||||
|
"bwhc-dto-java" to "0.3.0",
|
||||||
|
"hapi-fhir" to "6.10.5",
|
||||||
|
"httpclient5" to "5.2.3",
|
||||||
|
"mockito-kotlin" to "5.3.1",
|
||||||
|
// Webjars
|
||||||
|
"echarts" to "5.4.3",
|
||||||
|
"htmx.org" to "1.9.12"
|
||||||
|
)
|
||||||
|
|
||||||
java {
|
java {
|
||||||
sourceCompatibility = JavaVersion.VERSION_17
|
sourceCompatibility = JavaVersion.VERSION_21
|
||||||
}
|
}
|
||||||
|
|
||||||
sourceSets {
|
sourceSets {
|
||||||
@ -47,15 +57,22 @@ dependencies {
|
|||||||
implementation("org.springframework.boot:spring-boot-starter-thymeleaf")
|
implementation("org.springframework.boot:spring-boot-starter-thymeleaf")
|
||||||
implementation("org.springframework.boot:spring-boot-starter-web")
|
implementation("org.springframework.boot:spring-boot-starter-web")
|
||||||
implementation("org.springframework.boot:spring-boot-starter-data-jdbc")
|
implementation("org.springframework.boot:spring-boot-starter-data-jdbc")
|
||||||
|
implementation("org.springframework.boot:spring-boot-starter-security")
|
||||||
|
implementation("org.springframework.boot:spring-boot-starter-oauth2-client")
|
||||||
|
implementation("org.thymeleaf.extras:thymeleaf-extras-springsecurity6")
|
||||||
implementation("com.fasterxml.jackson.module:jackson-module-kotlin")
|
implementation("com.fasterxml.jackson.module:jackson-module-kotlin")
|
||||||
implementation("org.springframework.kafka:spring-kafka")
|
implementation("org.springframework.kafka:spring-kafka")
|
||||||
implementation("org.flywaydb:flyway-mysql")
|
implementation("org.flywaydb:flyway-mysql")
|
||||||
implementation("commons-codec:commons-codec")
|
implementation("commons-codec:commons-codec")
|
||||||
implementation("io.projectreactor.kotlin:reactor-kotlin-extensions")
|
implementation("io.projectreactor.kotlin:reactor-kotlin-extensions")
|
||||||
implementation("de.ukw.ccc:bwhc-dto-java:0.2.0")
|
implementation("de.ukw.ccc:bwhc-dto-java:${versions["bwhc-dto-java"]}")
|
||||||
implementation("ca.uhn.hapi.fhir:hapi-fhir-base:6.6.2")
|
implementation("ca.uhn.hapi.fhir:hapi-fhir-base:${versions["hapi-fhir"]}")
|
||||||
implementation("ca.uhn.hapi.fhir:hapi-fhir-structures-r4:6.6.2")
|
implementation("ca.uhn.hapi.fhir:hapi-fhir-structures-r4:${versions["hapi-fhir"]}")
|
||||||
implementation("org.apache.httpcomponents.client5:httpclient5:5.2.1")
|
implementation("org.apache.httpcomponents.client5:httpclient5:${versions["httpclient5"]}")
|
||||||
|
implementation("com.jayway.jsonpath:json-path")
|
||||||
|
implementation("org.webjars:webjars-locator:0.52")
|
||||||
|
implementation("org.webjars.npm:echarts:${versions["echarts"]}")
|
||||||
|
implementation("org.webjars.npm:htmx.org:${versions["htmx.org"]}")
|
||||||
runtimeOnly("org.mariadb.jdbc:mariadb-java-client")
|
runtimeOnly("org.mariadb.jdbc:mariadb-java-client")
|
||||||
runtimeOnly("org.postgresql:postgresql")
|
runtimeOnly("org.postgresql:postgresql")
|
||||||
developmentOnly("org.springframework.boot:spring-boot-devtools")
|
developmentOnly("org.springframework.boot:spring-boot-devtools")
|
||||||
@ -63,16 +80,19 @@ dependencies {
|
|||||||
annotationProcessor("org.springframework.boot:spring-boot-configuration-processor")
|
annotationProcessor("org.springframework.boot:spring-boot-configuration-processor")
|
||||||
providedRuntime("org.springframework.boot:spring-boot-starter-tomcat")
|
providedRuntime("org.springframework.boot:spring-boot-starter-tomcat")
|
||||||
testImplementation("org.springframework.boot:spring-boot-starter-test")
|
testImplementation("org.springframework.boot:spring-boot-starter-test")
|
||||||
|
testImplementation("org.springframework.security:spring-security-test")
|
||||||
testImplementation("io.projectreactor:reactor-test")
|
testImplementation("io.projectreactor:reactor-test")
|
||||||
testImplementation("org.mockito.kotlin:mockito-kotlin:5.0.0")
|
testImplementation("org.mockito.kotlin:mockito-kotlin:${versions["mockito-kotlin"]}")
|
||||||
integrationTestImplementation("org.testcontainers:junit-jupiter")
|
integrationTestImplementation("org.testcontainers:junit-jupiter")
|
||||||
integrationTestImplementation("org.testcontainers:postgresql")
|
integrationTestImplementation("org.testcontainers:postgresql")
|
||||||
|
// Override dependency version from org.testcontainers:junit-jupiter - CVE-2024-26308, CVE-2024-25710
|
||||||
|
integrationTestImplementation("org.apache.commons:commons-compress:1.26.2")
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks.withType<KotlinCompile> {
|
tasks.withType<KotlinCompile> {
|
||||||
kotlinOptions {
|
kotlinOptions {
|
||||||
freeCompilerArgs += "-Xjsr305=strict"
|
freeCompilerArgs += "-Xjsr305=strict"
|
||||||
jvmTarget = "17"
|
jvmTarget = "21"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -95,8 +115,15 @@ task<Test>("integrationTest") {
|
|||||||
tasks.named<BootBuildImage>("bootBuildImage") {
|
tasks.named<BootBuildImage>("bootBuildImage") {
|
||||||
imageName.set("ghcr.io/ccc-mf/etl-processor")
|
imageName.set("ghcr.io/ccc-mf/etl-processor")
|
||||||
|
|
||||||
|
// Binding for CA Certs
|
||||||
|
bindings.set(listOf(
|
||||||
|
"$rootDir/bindings/ca-certificates/:/platform/bindings/ca-certificates"
|
||||||
|
))
|
||||||
|
|
||||||
environment.set(environment.get() + mapOf(
|
environment.set(environment.get() + mapOf(
|
||||||
"BP_OCI_SOURCE" to "https://github.com/CCC-MF/etl-processor",
|
// Enable this line to embed CA Certs into image on build time
|
||||||
|
//"BP_EMBED_CERTS" to "true",
|
||||||
|
"BP_OCI_SOURCE" to "https://github.com/pcvolkmer/etl-processor",
|
||||||
"BP_OCI_LICENSES" to "AGPLv3",
|
"BP_OCI_LICENSES" to "AGPLv3",
|
||||||
"BP_OCI_DESCRIPTION" to "ETL Processor for bwHC MTB files"
|
"BP_OCI_DESCRIPTION" to "ETL Processor for bwHC MTB files"
|
||||||
))
|
))
|
||||||
|
57
deploy/docker-compose.yaml
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
|
||||||
|
|
||||||
|
services:
|
||||||
|
dnpm-etl-processor:
|
||||||
|
image: ghcr.io/ccc-mf/etl-processor:latest
|
||||||
|
environment:
|
||||||
|
LOGGING_LEVEL_DEV: ${DNPM_LOG_LEVEL:-INFO}
|
||||||
|
SPRING_KAFKA_SECURITY_PROTOCOL: ${DNPM_KAFKA_SECURITY_PROTOCOL:-SSL}
|
||||||
|
SPRING_KAFKA_SSL_TRUST-STORE-TYPE: PKCS12
|
||||||
|
SPRING_KAFKA_SSL_TRUST-STORE-LOCATION: /opt/dnpm-processor/ssl/truststore.jks
|
||||||
|
SPRING_KAFKA_SSL_TRUST-STORE-PASSWORD: ${KAFKA_TRUST_STORE_PASSWORD}
|
||||||
|
SPRING_KAFKA_SSL_KEY-STORE-TYPE: PKCS12
|
||||||
|
SPRING_KAFKA_SSL_KEY-STORE-LOCATION: /opt/dnpm-processor/ssl/keystore.jks
|
||||||
|
SPRING_KAFKA_SSL_KEY-STORE-PASSWORD: ${DNPM_PROCESSOR_KEY_STORE_PASSWORD}
|
||||||
|
SPRING_KAFKA_PRODUCER_COMPRESSION-TYPE: gzip
|
||||||
|
APP_KAFKA_TOPIC: ${DNPM_KAFKA_TOPIC}
|
||||||
|
APP_KAFKA_SERVERS: ${KAFKA_BROKERS}
|
||||||
|
APP_KAFKA_GROUP_ID: ${DNPM_KAFKA_GROUP_ID}
|
||||||
|
APP_KAFKA_RESPONSE_TOPIC: ${DNPM_KAFKA_RESPONSE_TOPIC}
|
||||||
|
APP_REST_URI: ${DNPM_BWHC_REST_URI}
|
||||||
|
APP_SECURITY_ADMIN_USER: ${DNPM_ADMIN_USER}
|
||||||
|
APP_SECURITY_ADMIN_PASSWORD: ${DNPM_ADMIN_PASSWORD}
|
||||||
|
SPRING_DATASOURCE_URL: ${DNPM_DATASOURCE_URL}
|
||||||
|
SPRING_DATASOURCE_PASSWORD: ${DNPM_MARIADB_USER_PW}
|
||||||
|
SPRING_DATASOURCE_USERNAME: ${DNPM_MARIADB_DB}
|
||||||
|
APP_PSEUDONYMIZE_GPAS_SSLCALOCATION: /workspace/opt/dnpm-processor/ssl/mosaic.crt
|
||||||
|
APP_PSEUDONYMIZE_GPAS_PASSWORD: ${DNPM_PSEUDONYMIZE_GPAS_PASSWORD}
|
||||||
|
APP_PSEUDONYMIZE_GPAS_USERNAME: ${DNPM_PSEUDONYMIZE_GPAS_USERNAME}
|
||||||
|
APP_PSEUDONYMIZE_GPAS_TARGET: ${DNPM_PSEUDONYMIZE_GPAS_TARGET}
|
||||||
|
APP_PSEUDONYMIZE_GPAS_URI: ${DNPM_PSEUDONYMIZE_GPAS_URI}
|
||||||
|
APP_PSEUDONYMIZE_PREFIX: ${DNPM_APP_PSEUDONYMIZE_PREFIX}
|
||||||
|
APP_PSEUDONYMIZER: ${DNPM_PSEUDONYMIZE_GENERATOR}
|
||||||
|
volumes:
|
||||||
|
- /etc/localtime:/etc/localtime:ro
|
||||||
|
- /etc/timezone:/etc/timezone:ro
|
||||||
|
#- ${DNPM_TO_SSL_KEYSTORE_LOCATION}:/workspace/opt/dnpm-processor/ssl/keystore.jks:ro
|
||||||
|
#- ${KAFKA_TRUST_STORE_LOCATION}:/workspace/opt/dnpm-processor/ssl/truststore.jks:ro
|
||||||
|
#- ${DNPM_PSEUDONYMIZE_GPAS_SSLCALOCATION}:/workspace/opt/dnpm-processor/ssl/mosaic.crt
|
||||||
|
|
||||||
|
depends_on:
|
||||||
|
- dnpm-monitor-db
|
||||||
|
ports:
|
||||||
|
- "${DNPM_MONITORING_HTTP_PORT:-8080}:8080"
|
||||||
|
|
||||||
|
# todo add volume
|
||||||
|
dnpm-monitor-db:
|
||||||
|
image: mariadb:10
|
||||||
|
environment:
|
||||||
|
MARIADB_DATABASE: ${DNPM_MARIADB_DB}
|
||||||
|
MARIADB_USER: ${DNPM_MARIADB_USER}
|
||||||
|
MARIADB_PASSWORD: ${DNPM_MARIADB_USER_PW}
|
||||||
|
MARIADB_ROOT_PASSWORD: ${DNPM_MARIADB_ROOT_PW}
|
||||||
|
expose:
|
||||||
|
- "3306"
|
||||||
|
|
||||||
|
|
||||||
|
|
44
deploy/env-sample.env
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
# monitoring access port
|
||||||
|
DNPM_MONITORING_HTTP_PORT=8088
|
||||||
|
DNPM_LOG_LEVEL=INFO
|
||||||
|
|
||||||
|
# ADMIN USER CREDENTIALS
|
||||||
|
DNPM_ADMIN_USER=admin
|
||||||
|
DNPM_ADMIN_PASSWORD=
|
||||||
|
|
||||||
|
# GPAS or BUILDIN
|
||||||
|
DNPM_PSEUDONYMIZE_GENERATOR=BUILDIN
|
||||||
|
DNPM_APP_PSEUDONYMIZE_PREFIX=ANONYM
|
||||||
|
DNPM_PSEUDONYMIZE_GPAS_URI=
|
||||||
|
DNPM_PSEUDONYMIZE_GPAS_TARGET=
|
||||||
|
DNPM_PSEUDONYMIZE_GPAS_USERNAME=
|
||||||
|
DNPM_PSEUDONYMIZE_GPAS_PASSWORD=
|
||||||
|
|
||||||
|
# path to ca root cert if needed
|
||||||
|
DNPM_PSEUDONYMIZE_GPAS_SSLCALOCATION=
|
||||||
|
|
||||||
|
DNPM_MARIADB_DB=dnpm_monitoring
|
||||||
|
DNPM_MARIADB_USER=$DNPM_MARIADB_DB
|
||||||
|
DNPM_MARIADB_USER_PW=MySuperSecurePassword111
|
||||||
|
DNPM_MARIADB_ROOT_PW=MySuperDuperSecurePassword111
|
||||||
|
|
||||||
|
# monitoring data db
|
||||||
|
DNPM_DATASOURCE_URL=jdbc:mariadb://dnpm-monitor-db:3306/$DNPM_MARIADB_DB
|
||||||
|
|
||||||
|
## TARGET SYSTEMS CONFIG
|
||||||
|
# in case of direct access to bwhc enter endpoint url here
|
||||||
|
DNPM_BWHC_REST_URI=
|
||||||
|
|
||||||
|
# produce mtb files to this topic - values 'false' disabling kafka processing
|
||||||
|
DNPM_KAFKA_TOPIC=false
|
||||||
|
KAFKA_BROKERS=false
|
||||||
|
DNPM_KAFKA_SECURITY_PROTOCOL=PLAINTEXT
|
||||||
|
|
||||||
|
# here we receive responses from bwhc
|
||||||
|
DNPM_KAFKA_RESPONSE_TOPIC=dnpm-response
|
||||||
|
DNPM_KAFKA_GROUP_ID=dnpm
|
||||||
|
|
||||||
|
# SSL or PLAINTEXT
|
||||||
|
DNPM_PROCESSOR_KEY_STORE_PASSWORD=
|
||||||
|
DNPM_TO_SSL_KEYSTORE_LOCATION=
|
||||||
|
|
@ -4,8 +4,33 @@ services:
|
|||||||
hostname: kafka
|
hostname: kafka
|
||||||
ports:
|
ports:
|
||||||
- "9092:9092"
|
- "9092:9092"
|
||||||
|
- "9094:9094"
|
||||||
environment:
|
environment:
|
||||||
ALLOW_PLAINTEXT_LISTENER: "yes"
|
ALLOW_PLAINTEXT_LISTENER: "yes"
|
||||||
|
KAFKA_CFG_NODE_ID: "0"
|
||||||
|
KAFKA_CFG_PROCESS_ROLES: "controller,broker"
|
||||||
|
KAFKA_CFG_LISTENERS: PLAINTEXT://:9092,CONTROLLER://:9093,EXTERNAL://:9094
|
||||||
|
KAFKA_CFG_ADVERTISED_LISTENERS: PLAINTEXT://kafka:9092,EXTERNAL://localhost:9094
|
||||||
|
KAFKA_CFG_LISTENER_SECURITY_PROTOCOL_MAP: CONTROLLER:PLAINTEXT,EXTERNAL:PLAINTEXT,PLAINTEXT:PLAINTEXT
|
||||||
|
KAFKA_CFG_INTER_BROKER_LISTENER_NAME: PLAINTEXT
|
||||||
|
KAFKA_CFG_AUTO_CREATE_TOPICS_ENABLE: true
|
||||||
|
KAFKA_CFG_CONTROLLER_QUORUM_VOTERS: 0@kafka:9093
|
||||||
|
KAFKA_CFG_CONTROLLER_LISTENER_NAMES: CONTROLLER
|
||||||
|
|
||||||
|
akhq:
|
||||||
|
image: tchiotludo/akhq:0.21.0
|
||||||
|
environment:
|
||||||
|
AKHQ_CONFIGURATION: |
|
||||||
|
akhq:
|
||||||
|
connections:
|
||||||
|
docker-kafka-server:
|
||||||
|
properties:
|
||||||
|
bootstrap.servers: "kafka:9092"
|
||||||
|
connect:
|
||||||
|
- name: "kafka-connect"
|
||||||
|
url: "http://kafka-connect:8083"
|
||||||
|
ports:
|
||||||
|
- "8084:8080"
|
||||||
|
|
||||||
mariadb:
|
mariadb:
|
||||||
image: mariadb:10
|
image: mariadb:10
|
||||||
@ -16,6 +41,7 @@ services:
|
|||||||
MARIADB_USER: dev
|
MARIADB_USER: dev
|
||||||
MARIADB_PASSWORD: dev
|
MARIADB_PASSWORD: dev
|
||||||
MARIADB_ROOT_PASSWORD: dev
|
MARIADB_ROOT_PASSWORD: dev
|
||||||
|
|
||||||
# postgres:
|
# postgres:
|
||||||
# image: postgres:alpine
|
# image: postgres:alpine
|
||||||
# ports:
|
# ports:
|
||||||
@ -23,4 +49,4 @@ services:
|
|||||||
# environment:
|
# environment:
|
||||||
# POSTGRES_DB: dev
|
# POSTGRES_DB: dev
|
||||||
# POSTGRES_USER: dev
|
# POSTGRES_USER: dev
|
||||||
# POSTGRES_PASSWORD: dev
|
# POSTGRES_PASSWORD: dev
|
28
docs/docker-compose.yml
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
### Example for docker-compose
|
||||||
|
version: '3.7'
|
||||||
|
|
||||||
|
volumes:
|
||||||
|
data:
|
||||||
|
|
||||||
|
services:
|
||||||
|
|
||||||
|
### ETL-Processor
|
||||||
|
etl-processor:
|
||||||
|
image: ghcr.io/ccc-mf/etl-processor:latest
|
||||||
|
ports:
|
||||||
|
- 8080:8080
|
||||||
|
environment:
|
||||||
|
APP_REST_URI: http://bwhc-backend/bwhc/etl/api
|
||||||
|
SPRING_DATASOURCE_URL: jdbc:postgresql://postgres/etl
|
||||||
|
SPRING_DATASOURCE_USERNAME: etl
|
||||||
|
SPRING_DATASOURCE_PASSWORD: etl-password
|
||||||
|
|
||||||
|
### Database
|
||||||
|
postgres:
|
||||||
|
image: postgres:alpine
|
||||||
|
environment:
|
||||||
|
POSTGRES_DB: etl
|
||||||
|
POSTGRES_USER: etl
|
||||||
|
POSTGRES_PASSWORD: etl-password
|
||||||
|
volumes:
|
||||||
|
- data:/var/lib/postgresql/data
|
BIN
docs/etl.png
Normal file
After Width: | Height: | Size: 120 KiB |
BIN
docs/login.png
Normal file
After Width: | Height: | Size: 9.2 KiB |
BIN
docs/tokens.png
Normal file
After Width: | Height: | Size: 30 KiB |
BIN
docs/userroles.png
Normal file
After Width: | Height: | Size: 16 KiB |
2
gradle/wrapper/gradle-wrapper.properties
vendored
@ -1,6 +1,6 @@
|
|||||||
distributionBase=GRADLE_USER_HOME
|
distributionBase=GRADLE_USER_HOME
|
||||||
distributionPath=wrapper/dists
|
distributionPath=wrapper/dists
|
||||||
distributionUrl=https\://services.gradle.org/distributions/gradle-8.1.1-bin.zip
|
distributionUrl=https\://services.gradle.org/distributions/gradle-8.6-bin.zip
|
||||||
networkTimeout=10000
|
networkTimeout=10000
|
||||||
zipStoreBase=GRADLE_USER_HOME
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
zipStorePath=wrapper/dists
|
zipStorePath=wrapper/dists
|
||||||
|
@ -19,22 +19,127 @@
|
|||||||
|
|
||||||
package dev.dnpm.etl.processor
|
package dev.dnpm.etl.processor
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import de.ukw.ccc.bwhc.dto.*
|
||||||
|
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
||||||
|
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
||||||
import dev.dnpm.etl.processor.output.MtbFileSender
|
import dev.dnpm.etl.processor.output.MtbFileSender
|
||||||
|
import org.assertj.core.api.Assertions.assertThat
|
||||||
|
import org.junit.jupiter.api.BeforeEach
|
||||||
|
import org.junit.jupiter.api.Nested
|
||||||
import org.junit.jupiter.api.Test
|
import org.junit.jupiter.api.Test
|
||||||
import org.junit.jupiter.api.extension.ExtendWith
|
import org.junit.jupiter.api.extension.ExtendWith
|
||||||
|
import org.mockito.kotlin.*
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired
|
||||||
|
import org.springframework.boot.test.autoconfigure.web.servlet.AutoConfigureMockMvc
|
||||||
import org.springframework.boot.test.context.SpringBootTest
|
import org.springframework.boot.test.context.SpringBootTest
|
||||||
import org.springframework.boot.test.mock.mockito.MockBean
|
import org.springframework.boot.test.mock.mockito.MockBean
|
||||||
|
import org.springframework.context.ApplicationContext
|
||||||
|
import org.springframework.http.MediaType
|
||||||
|
import org.springframework.test.context.TestPropertySource
|
||||||
import org.springframework.test.context.junit.jupiter.SpringExtension
|
import org.springframework.test.context.junit.jupiter.SpringExtension
|
||||||
|
import org.springframework.test.web.servlet.MockMvc
|
||||||
|
import org.springframework.test.web.servlet.post
|
||||||
import org.testcontainers.junit.jupiter.Testcontainers
|
import org.testcontainers.junit.jupiter.Testcontainers
|
||||||
|
|
||||||
@Testcontainers
|
@Testcontainers
|
||||||
@ExtendWith(SpringExtension::class)
|
@ExtendWith(SpringExtension::class)
|
||||||
@SpringBootTest
|
@SpringBootTest
|
||||||
@MockBean(MtbFileSender::class)
|
@MockBean(MtbFileSender::class)
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.rest.uri=http://example.com",
|
||||||
|
"app.pseudonymize.generator=buildin"
|
||||||
|
]
|
||||||
|
)
|
||||||
class EtlProcessorApplicationTests : AbstractTestcontainerTest() {
|
class EtlProcessorApplicationTests : AbstractTestcontainerTest() {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun contextLoadsIfMtbFileSenderConfigured() {
|
fun contextLoadsIfMtbFileSenderConfigured(@Autowired context: ApplicationContext) {
|
||||||
|
// Simply check bean configuration
|
||||||
|
assertThat(context).isNotNull
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.MOCK)
|
||||||
|
@AutoConfigureMockMvc
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=buildin",
|
||||||
|
"app.transformations[0].path=diagnoses[*].icd10.version",
|
||||||
|
"app.transformations[0].from=2013",
|
||||||
|
"app.transformations[0].to=2014",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
inner class TransformationTest {
|
||||||
|
|
||||||
|
@MockBean
|
||||||
|
private lateinit var mtbFileSender: MtbFileSender
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private lateinit var mockMvc: MockMvc
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private lateinit var objectMapper: ObjectMapper
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
fun setup(@Autowired requestRepository: RequestRepository) {
|
||||||
|
requestRepository.deleteAll()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun mtbFileIsTransformed() {
|
||||||
|
doAnswer {
|
||||||
|
MtbFileSender.Response(RequestStatus.SUCCESS)
|
||||||
|
}.whenever(mtbFileSender).send(any<MtbFileSender.MtbFileRequest>())
|
||||||
|
|
||||||
|
val mtbFile = MtbFile.builder()
|
||||||
|
.withPatient(
|
||||||
|
Patient.builder()
|
||||||
|
.withId("TEST_12345678")
|
||||||
|
.withBirthDate("2000-08-08")
|
||||||
|
.withGender(Patient.Gender.MALE)
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withConsent(
|
||||||
|
Consent.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withStatus(Consent.Status.ACTIVE)
|
||||||
|
.withPatient("TEST_12345678")
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withEpisode(
|
||||||
|
Episode.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withPatient("TEST_12345678")
|
||||||
|
.withPeriod(PeriodStart("2023-08-08"))
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withDiagnoses(
|
||||||
|
listOf(
|
||||||
|
Diagnosis.builder()
|
||||||
|
.withId("1234")
|
||||||
|
.withIcd10(Icd10.builder().withCode("F79.9").withVersion("2013").build())
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.build()
|
||||||
|
|
||||||
|
mockMvc.post("/mtbfile") {
|
||||||
|
content = objectMapper.writeValueAsString(mtbFile)
|
||||||
|
contentType = MediaType.APPLICATION_JSON
|
||||||
|
}.andExpect {
|
||||||
|
status {
|
||||||
|
isAccepted()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val captor = argumentCaptor<MtbFileSender.MtbFileRequest>()
|
||||||
|
verify(mtbFileSender).send(captor.capture())
|
||||||
|
assertThat(captor.firstValue.mtbFile.diagnoses).hasSize(1).allMatch { diagnosis ->
|
||||||
|
diagnosis.icd10.version == "2014"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -20,9 +20,15 @@
|
|||||||
package dev.dnpm.etl.processor.config
|
package dev.dnpm.etl.processor.config
|
||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import dev.dnpm.etl.processor.input.KafkaInputListener
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
||||||
import dev.dnpm.etl.processor.output.KafkaMtbFileSender
|
import dev.dnpm.etl.processor.output.KafkaMtbFileSender
|
||||||
import dev.dnpm.etl.processor.output.RestMtbFileSender
|
import dev.dnpm.etl.processor.output.RestMtbFileSender
|
||||||
|
import dev.dnpm.etl.processor.pseudonym.AnonymizingGenerator
|
||||||
|
import dev.dnpm.etl.processor.pseudonym.GpasPseudonymGenerator
|
||||||
|
import dev.dnpm.etl.processor.services.RequestProcessor
|
||||||
|
import dev.dnpm.etl.processor.services.TokenRepository
|
||||||
|
import dev.dnpm.etl.processor.services.TokenService
|
||||||
import org.assertj.core.api.Assertions.assertThat
|
import org.assertj.core.api.Assertions.assertThat
|
||||||
import org.junit.jupiter.api.Nested
|
import org.junit.jupiter.api.Nested
|
||||||
import org.junit.jupiter.api.Test
|
import org.junit.jupiter.api.Test
|
||||||
@ -33,11 +39,26 @@ import org.springframework.boot.test.context.SpringBootTest
|
|||||||
import org.springframework.boot.test.mock.mockito.MockBean
|
import org.springframework.boot.test.mock.mockito.MockBean
|
||||||
import org.springframework.boot.test.mock.mockito.MockBeans
|
import org.springframework.boot.test.mock.mockito.MockBeans
|
||||||
import org.springframework.context.ApplicationContext
|
import org.springframework.context.ApplicationContext
|
||||||
|
import org.springframework.retry.support.RetryTemplate
|
||||||
|
import org.springframework.security.crypto.password.PasswordEncoder
|
||||||
|
import org.springframework.security.provisioning.InMemoryUserDetailsManager
|
||||||
import org.springframework.test.context.ContextConfiguration
|
import org.springframework.test.context.ContextConfiguration
|
||||||
import org.springframework.test.context.TestPropertySource
|
import org.springframework.test.context.TestPropertySource
|
||||||
|
|
||||||
@SpringBootTest
|
@SpringBootTest
|
||||||
@ContextConfiguration(classes = [KafkaAutoConfiguration::class, AppKafkaConfiguration::class, AppRestConfiguration::class])
|
@ContextConfiguration(classes = [
|
||||||
|
AppConfiguration::class,
|
||||||
|
AppSecurityConfiguration::class,
|
||||||
|
KafkaAutoConfiguration::class,
|
||||||
|
AppKafkaConfiguration::class,
|
||||||
|
AppRestConfiguration::class
|
||||||
|
])
|
||||||
|
@MockBean(ObjectMapper::class)
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=BUILDIN",
|
||||||
|
]
|
||||||
|
)
|
||||||
class AppConfigurationTest {
|
class AppConfigurationTest {
|
||||||
|
|
||||||
@Nested
|
@Nested
|
||||||
@ -60,15 +81,12 @@ class AppConfigurationTest {
|
|||||||
@TestPropertySource(
|
@TestPropertySource(
|
||||||
properties = [
|
properties = [
|
||||||
"app.kafka.servers=localhost:9092",
|
"app.kafka.servers=localhost:9092",
|
||||||
"app.kafka.topic=test",
|
"app.kafka.output-topic=test",
|
||||||
"app.kafka.response-topic=test-response",
|
"app.kafka.output-response-topic=test-response",
|
||||||
"app.kafka.group-id=test"
|
"app.kafka.group-id=test"
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
@MockBeans(value = [
|
@MockBean(RequestRepository::class)
|
||||||
MockBean(ObjectMapper::class),
|
|
||||||
MockBean(RequestRepository::class)
|
|
||||||
])
|
|
||||||
inner class AppConfigurationKafkaTest(private val context: ApplicationContext) {
|
inner class AppConfigurationKafkaTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -84,8 +102,8 @@ class AppConfigurationTest {
|
|||||||
properties = [
|
properties = [
|
||||||
"app.rest.uri=http://localhost:9000",
|
"app.rest.uri=http://localhost:9000",
|
||||||
"app.kafka.servers=localhost:9092",
|
"app.kafka.servers=localhost:9092",
|
||||||
"app.kafka.topic=test",
|
"app.kafka.output-topic=test",
|
||||||
"app.kafka.response-topic=test-response",
|
"app.kafka.output-response-topic=test-response",
|
||||||
"app.kafka.group-id=test"
|
"app.kafka.group-id=test"
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
@ -99,4 +117,192 @@ class AppConfigurationTest {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.kafka.servers=localhost:9092",
|
||||||
|
"app.kafka.output-topic=test",
|
||||||
|
"app.kafka.output-response-topic=test-response",
|
||||||
|
"app.kafka.group-id=test"
|
||||||
|
]
|
||||||
|
)
|
||||||
|
inner class AppConfigurationWithoutKafkaInputTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldNotUseKafkaInputListener() {
|
||||||
|
assertThrows<NoSuchBeanDefinitionException> { context.getBean(KafkaInputListener::class.java) }
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.kafka.servers=localhost:9092",
|
||||||
|
"app.kafka.input-topic=test_input",
|
||||||
|
"app.kafka.output-topic=test",
|
||||||
|
"app.kafka.output-response-topic=test-response",
|
||||||
|
"app.kafka.group-id=test"
|
||||||
|
]
|
||||||
|
)
|
||||||
|
@MockBean(RequestProcessor::class)
|
||||||
|
inner class AppConfigurationUsingKafkaInputTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldUseKafkaInputListener() {
|
||||||
|
assertThat(context.getBean(KafkaInputListener::class.java)).isNotNull
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.transformations[0].path=consent.status",
|
||||||
|
"app.transformations[0].from=rejected",
|
||||||
|
"app.transformations[0].to=accept",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
inner class AppConfigurationTransformationTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldRecognizeTransformations() {
|
||||||
|
val appConfigProperties = context.getBean(AppConfigProperties::class.java)
|
||||||
|
|
||||||
|
assertThat(appConfigProperties).isNotNull
|
||||||
|
assertThat(appConfigProperties.transformations).hasSize(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
inner class AppConfigurationPseudonymizeTest {
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=",
|
||||||
|
"app.pseudonymizer=buildin",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
inner class AppConfigurationPseudonymizerBuildinTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldUseConfiguredGenerator() {
|
||||||
|
assertThat(context.getBean(AnonymizingGenerator::class.java)).isNotNull
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=",
|
||||||
|
"app.pseudonymizer=gpas",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
inner class AppConfigurationPseudonymizerGpasTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldUseConfiguredGenerator() {
|
||||||
|
assertThat(context.getBean(GpasPseudonymGenerator::class.java)).isNotNull
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=buildin",
|
||||||
|
"app.pseudonymizer=",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
inner class AppConfigurationPseudonymizeGeneratorBuildinTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldUseConfiguredGenerator() {
|
||||||
|
assertThat(context.getBean(AnonymizingGenerator::class.java)).isNotNull
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=gpas",
|
||||||
|
"app.pseudonymizer=",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
inner class AppConfigurationPseudonymizeGeneratorGpasTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldUseConfiguredGenerator() {
|
||||||
|
assertThat(context.getBean(GpasPseudonymGenerator::class.java)).isNotNull
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.security.enable-tokens=true"
|
||||||
|
]
|
||||||
|
)
|
||||||
|
@MockBeans(value = [
|
||||||
|
MockBean(InMemoryUserDetailsManager::class),
|
||||||
|
MockBean(PasswordEncoder::class),
|
||||||
|
MockBean(TokenRepository::class)
|
||||||
|
])
|
||||||
|
inner class AppConfigurationTokenEnabledTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun checkTokenService() {
|
||||||
|
assertThat(context.getBean(TokenService::class.java)).isNotNull
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@MockBeans(value = [
|
||||||
|
MockBean(InMemoryUserDetailsManager::class),
|
||||||
|
MockBean(PasswordEncoder::class),
|
||||||
|
MockBean(TokenRepository::class)
|
||||||
|
])
|
||||||
|
inner class AppConfigurationTokenDisabledTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun checkTokenService() {
|
||||||
|
assertThrows<NoSuchBeanDefinitionException> { context.getBean(TokenService::class.java) }
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nested
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.rest.uri=http://localhost:9000",
|
||||||
|
"app.max-retry-attempts=5"
|
||||||
|
]
|
||||||
|
)
|
||||||
|
inner class AppConfigurationRetryTest(private val context: ApplicationContext) {
|
||||||
|
|
||||||
|
private val maxRetryAttempts = 5
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldUseRetryTemplateWithConfiguredMaxAttempts() {
|
||||||
|
val retryTemplate = context.getBean(RetryTemplate::class.java)
|
||||||
|
assertThat(retryTemplate).isNotNull
|
||||||
|
|
||||||
|
assertThrows<RuntimeException> {
|
||||||
|
retryTemplate.execute<Void, RuntimeException> {
|
||||||
|
assertThat(it.retryCount).isLessThan(maxRetryAttempts)
|
||||||
|
throw RuntimeException()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -0,0 +1,157 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.input
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import de.ukw.ccc.bwhc.dto.*
|
||||||
|
import dev.dnpm.etl.processor.config.AppSecurityConfiguration
|
||||||
|
import dev.dnpm.etl.processor.services.RequestProcessor
|
||||||
|
import dev.dnpm.etl.processor.services.TokenRepository
|
||||||
|
import org.junit.jupiter.api.BeforeEach
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith
|
||||||
|
import org.mockito.ArgumentMatchers.anyString
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension
|
||||||
|
import org.mockito.kotlin.any
|
||||||
|
import org.mockito.kotlin.never
|
||||||
|
import org.mockito.kotlin.times
|
||||||
|
import org.mockito.kotlin.verify
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired
|
||||||
|
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest
|
||||||
|
import org.springframework.boot.test.mock.mockito.MockBean
|
||||||
|
import org.springframework.http.MediaType
|
||||||
|
import org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.anonymous
|
||||||
|
import org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.user
|
||||||
|
import org.springframework.test.context.ContextConfiguration
|
||||||
|
import org.springframework.test.context.TestPropertySource
|
||||||
|
import org.springframework.test.context.junit.jupiter.SpringExtension
|
||||||
|
import org.springframework.test.web.servlet.MockMvc
|
||||||
|
import org.springframework.test.web.servlet.delete
|
||||||
|
import org.springframework.test.web.servlet.post
|
||||||
|
|
||||||
|
@WebMvcTest(controllers = [MtbFileRestController::class])
|
||||||
|
@ExtendWith(value = [MockitoExtension::class, SpringExtension::class])
|
||||||
|
@ContextConfiguration(
|
||||||
|
classes = [
|
||||||
|
MtbFileRestController::class,
|
||||||
|
AppSecurityConfiguration::class
|
||||||
|
]
|
||||||
|
)
|
||||||
|
@MockBean(TokenRepository::class, RequestProcessor::class)
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=BUILDIN",
|
||||||
|
"app.security.admin-user=admin",
|
||||||
|
"app.security.admin-password={noop}very-secret",
|
||||||
|
"app.security.enable-tokens=true"
|
||||||
|
]
|
||||||
|
)
|
||||||
|
class MtbFileRestControllerTest {
|
||||||
|
|
||||||
|
private lateinit var mockMvc: MockMvc
|
||||||
|
|
||||||
|
private lateinit var requestProcessor: RequestProcessor
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
fun setup(
|
||||||
|
@Autowired mockMvc: MockMvc,
|
||||||
|
@Autowired requestProcessor: RequestProcessor
|
||||||
|
) {
|
||||||
|
this.mockMvc = mockMvc
|
||||||
|
this.requestProcessor = requestProcessor
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testShouldGrantPermissionToSendMtbFile() {
|
||||||
|
mockMvc.post("/mtbfile") {
|
||||||
|
with(user("onkostarserver").roles("MTBFILE"))
|
||||||
|
contentType = MediaType.APPLICATION_JSON
|
||||||
|
content = ObjectMapper().writeValueAsString(mtbFile)
|
||||||
|
}.andExpect {
|
||||||
|
status { isAccepted() }
|
||||||
|
}
|
||||||
|
|
||||||
|
verify(requestProcessor, times(1)).processMtbFile(any())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testShouldDenyPermissionToSendMtbFile() {
|
||||||
|
mockMvc.post("/mtbfile") {
|
||||||
|
with(anonymous())
|
||||||
|
contentType = MediaType.APPLICATION_JSON
|
||||||
|
content = ObjectMapper().writeValueAsString(mtbFile)
|
||||||
|
}.andExpect {
|
||||||
|
status { isUnauthorized() }
|
||||||
|
}
|
||||||
|
|
||||||
|
verify(requestProcessor, never()).processMtbFile(any())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testShouldGrantPermissionToDeletePatientData() {
|
||||||
|
mockMvc.delete("/mtbfile/12345678") {
|
||||||
|
with(user("onkostarserver").roles("MTBFILE"))
|
||||||
|
}.andExpect {
|
||||||
|
status { isAccepted() }
|
||||||
|
}
|
||||||
|
|
||||||
|
verify(requestProcessor, times(1)).processDeletion(anyString())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testShouldDenyPermissionToDeletePatientData() {
|
||||||
|
mockMvc.delete("/mtbfile/12345678") {
|
||||||
|
with(anonymous())
|
||||||
|
}.andExpect {
|
||||||
|
status { isUnauthorized() }
|
||||||
|
}
|
||||||
|
|
||||||
|
verify(requestProcessor, never()).processDeletion(anyString())
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
|
||||||
|
val mtbFile: MtbFile = MtbFile.builder()
|
||||||
|
.withPatient(
|
||||||
|
Patient.builder()
|
||||||
|
.withId("PID")
|
||||||
|
.withBirthDate("2000-08-08")
|
||||||
|
.withGender(Patient.Gender.MALE)
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withConsent(
|
||||||
|
Consent.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withStatus(Consent.Status.ACTIVE)
|
||||||
|
.withPatient("PID")
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withEpisode(
|
||||||
|
Episode.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withPatient("PID")
|
||||||
|
.withPeriod(PeriodStart("2023-08-08"))
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.build()
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -32,6 +32,7 @@ import org.junit.jupiter.api.extension.ExtendWith
|
|||||||
import org.springframework.beans.factory.annotation.Autowired
|
import org.springframework.beans.factory.annotation.Autowired
|
||||||
import org.springframework.boot.test.context.SpringBootTest
|
import org.springframework.boot.test.context.SpringBootTest
|
||||||
import org.springframework.boot.test.mock.mockito.MockBean
|
import org.springframework.boot.test.mock.mockito.MockBean
|
||||||
|
import org.springframework.test.context.TestPropertySource
|
||||||
import org.springframework.test.context.junit.jupiter.SpringExtension
|
import org.springframework.test.context.junit.jupiter.SpringExtension
|
||||||
import org.springframework.transaction.annotation.Transactional
|
import org.springframework.transaction.annotation.Transactional
|
||||||
import org.testcontainers.junit.jupiter.Testcontainers
|
import org.testcontainers.junit.jupiter.Testcontainers
|
||||||
@ -43,6 +44,12 @@ import java.util.*
|
|||||||
@SpringBootTest
|
@SpringBootTest
|
||||||
@Transactional
|
@Transactional
|
||||||
@MockBean(MtbFileSender::class)
|
@MockBean(MtbFileSender::class)
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=buildin",
|
||||||
|
"app.rest.uri=http://example.com"
|
||||||
|
]
|
||||||
|
)
|
||||||
class RequestServiceIntegrationTest : AbstractTestcontainerTest() {
|
class RequestServiceIntegrationTest : AbstractTestcontainerTest() {
|
||||||
|
|
||||||
private lateinit var requestRepository: RequestRepository
|
private lateinit var requestRepository: RequestRepository
|
||||||
|
@ -0,0 +1,116 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.web
|
||||||
|
|
||||||
|
import dev.dnpm.etl.processor.config.AppConfiguration
|
||||||
|
import dev.dnpm.etl.processor.config.AppSecurityConfiguration
|
||||||
|
import dev.dnpm.etl.processor.monitoring.ConnectionCheckService
|
||||||
|
import dev.dnpm.etl.processor.monitoring.RestConnectionCheckService
|
||||||
|
import dev.dnpm.etl.processor.output.MtbFileSender
|
||||||
|
import dev.dnpm.etl.processor.pseudonym.Generator
|
||||||
|
import dev.dnpm.etl.processor.services.RequestProcessor
|
||||||
|
import dev.dnpm.etl.processor.services.TokenRepository
|
||||||
|
import dev.dnpm.etl.processor.services.TransformationService
|
||||||
|
import org.junit.jupiter.api.BeforeEach
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired
|
||||||
|
import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest
|
||||||
|
import org.springframework.boot.test.mock.mockito.MockBean
|
||||||
|
import org.springframework.http.HttpHeaders
|
||||||
|
import org.springframework.http.MediaType
|
||||||
|
import org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.anonymous
|
||||||
|
import org.springframework.security.test.web.servlet.request.SecurityMockMvcRequestPostProcessors.user
|
||||||
|
import org.springframework.test.context.ContextConfiguration
|
||||||
|
import org.springframework.test.context.TestPropertySource
|
||||||
|
import org.springframework.test.context.junit.jupiter.SpringExtension
|
||||||
|
import org.springframework.test.web.servlet.MockMvc
|
||||||
|
import org.springframework.test.web.servlet.get
|
||||||
|
import reactor.core.publisher.Sinks
|
||||||
|
|
||||||
|
abstract class MockSink : Sinks.Many<Boolean>
|
||||||
|
|
||||||
|
@WebMvcTest(controllers = [ConfigController::class])
|
||||||
|
@ExtendWith(value = [MockitoExtension::class, SpringExtension::class])
|
||||||
|
@ContextConfiguration(
|
||||||
|
classes = [
|
||||||
|
ConfigController::class,
|
||||||
|
AppConfiguration::class,
|
||||||
|
AppSecurityConfiguration::class
|
||||||
|
]
|
||||||
|
)
|
||||||
|
@TestPropertySource(
|
||||||
|
properties = [
|
||||||
|
"app.pseudonymize.generator=BUILDIN",
|
||||||
|
"app.security.admin-user=admin",
|
||||||
|
"app.security.admin-password={noop}very-secret",
|
||||||
|
"app.security.enable-tokens=true"
|
||||||
|
]
|
||||||
|
)
|
||||||
|
@MockBean(name = "configsUpdateProducer", classes = [MockSink::class])
|
||||||
|
@MockBean(
|
||||||
|
Generator::class,
|
||||||
|
MtbFileSender::class,
|
||||||
|
ConnectionCheckService::class,
|
||||||
|
RequestProcessor::class,
|
||||||
|
TransformationService::class,
|
||||||
|
TokenRepository::class,
|
||||||
|
RestConnectionCheckService::class
|
||||||
|
)
|
||||||
|
class ConfigControllerTest {
|
||||||
|
|
||||||
|
private lateinit var mockMvc: MockMvc
|
||||||
|
|
||||||
|
private lateinit var requestProcessor: RequestProcessor
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
fun setup(
|
||||||
|
@Autowired mockMvc: MockMvc,
|
||||||
|
@Autowired requestProcessor: RequestProcessor
|
||||||
|
) {
|
||||||
|
this.mockMvc = mockMvc
|
||||||
|
this.requestProcessor = requestProcessor
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testShouldShowConfigPageIfLoggedIn() {
|
||||||
|
mockMvc.get("/configs") {
|
||||||
|
with(user("admin").roles("ADMIN"))
|
||||||
|
accept(MediaType.TEXT_HTML)
|
||||||
|
}.andExpect {
|
||||||
|
status { isOk() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testShouldRedirectToLoginPageIfNotLoggedIn() {
|
||||||
|
mockMvc.get("/configs") {
|
||||||
|
with(anonymous())
|
||||||
|
accept(MediaType.TEXT_HTML)
|
||||||
|
}.andExpect {
|
||||||
|
status { isFound() }
|
||||||
|
header {
|
||||||
|
stringValues(HttpHeaders.LOCATION, "http://localhost/login")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -41,14 +41,7 @@ import org.slf4j.Logger;
|
|||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.http.*;
|
import org.springframework.http.*;
|
||||||
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
|
import org.springframework.http.client.HttpComponentsClientHttpRequestFactory;
|
||||||
import org.springframework.retry.RetryCallback;
|
|
||||||
import org.springframework.retry.RetryContext;
|
|
||||||
import org.springframework.retry.RetryListener;
|
|
||||||
import org.springframework.retry.RetryPolicy;
|
|
||||||
import org.springframework.retry.backoff.ExponentialBackOffPolicy;
|
|
||||||
import org.springframework.retry.policy.SimpleRetryPolicy;
|
|
||||||
import org.springframework.retry.support.RetryTemplate;
|
import org.springframework.retry.support.RetryTemplate;
|
||||||
import org.springframework.web.client.RestClientException;
|
|
||||||
import org.springframework.web.client.RestTemplate;
|
import org.springframework.web.client.RestTemplate;
|
||||||
|
|
||||||
import javax.net.ssl.SSLContext;
|
import javax.net.ssl.SSLContext;
|
||||||
@ -56,7 +49,6 @@ import javax.net.ssl.TrustManagerFactory;
|
|||||||
import java.io.BufferedInputStream;
|
import java.io.BufferedInputStream;
|
||||||
import java.io.FileInputStream;
|
import java.io.FileInputStream;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.ConnectException;
|
|
||||||
import java.security.KeyManagementException;
|
import java.security.KeyManagementException;
|
||||||
import java.security.KeyStore;
|
import java.security.KeyStore;
|
||||||
import java.security.KeyStoreException;
|
import java.security.KeyStoreException;
|
||||||
@ -65,23 +57,23 @@ import java.security.cert.CertificateException;
|
|||||||
import java.security.cert.CertificateFactory;
|
import java.security.cert.CertificateFactory;
|
||||||
import java.security.cert.X509Certificate;
|
import java.security.cert.X509Certificate;
|
||||||
import java.util.Base64;
|
import java.util.Base64;
|
||||||
import java.util.HashMap;
|
|
||||||
|
|
||||||
public class GpasPseudonymGenerator implements Generator {
|
public class GpasPseudonymGenerator implements Generator {
|
||||||
|
|
||||||
|
private final static FhirContext r4Context = FhirContext.forR4();
|
||||||
private final String gPasUrl;
|
private final String gPasUrl;
|
||||||
private final String psnTargetDomain;
|
private final String psnTargetDomain;
|
||||||
private static FhirContext r4Context = FhirContext.forR4();
|
|
||||||
private final HttpHeaders httpHeader;
|
private final HttpHeaders httpHeader;
|
||||||
|
private final RetryTemplate retryTemplate;
|
||||||
private final RetryTemplate retryTemplate = defaultTemplate();
|
|
||||||
|
|
||||||
private final Logger log = LoggerFactory.getLogger(GpasPseudonymGenerator.class);
|
private final Logger log = LoggerFactory.getLogger(GpasPseudonymGenerator.class);
|
||||||
|
|
||||||
private SSLContext customSslContext;
|
private final RestTemplate restTemplate;
|
||||||
private RestTemplate restTemplate;
|
|
||||||
|
|
||||||
public GpasPseudonymGenerator(GPasConfigProperties gpasCfg) {
|
private SSLContext customSslContext;
|
||||||
|
|
||||||
|
public GpasPseudonymGenerator(GPasConfigProperties gpasCfg, RetryTemplate retryTemplate) {
|
||||||
|
this.retryTemplate = retryTemplate;
|
||||||
|
this.restTemplate = getRestTemplete();
|
||||||
|
|
||||||
this.gPasUrl = gpasCfg.getUri();
|
this.gPasUrl = gpasCfg.getUri();
|
||||||
this.psnTargetDomain = gpasCfg.getTarget();
|
this.psnTargetDomain = gpasCfg.getTarget();
|
||||||
@ -90,12 +82,16 @@ public class GpasPseudonymGenerator implements Generator {
|
|||||||
try {
|
try {
|
||||||
if (StringUtils.isNotBlank(gpasCfg.getSslCaLocation())) {
|
if (StringUtils.isNotBlank(gpasCfg.getSslCaLocation())) {
|
||||||
customSslContext = getSslContext(gpasCfg.getSslCaLocation());
|
customSslContext = getSslContext(gpasCfg.getSslCaLocation());
|
||||||
|
log.warn(String.format("%s has been initialized with SSL certificate %s. This is deprecated in favor of including Root CA.",
|
||||||
|
this.getClass().getName(), gpasCfg.getSslCaLocation()));
|
||||||
}
|
}
|
||||||
} catch (IOException | KeyManagementException | KeyStoreException | CertificateException |
|
} catch (IOException | KeyManagementException | KeyStoreException | CertificateException |
|
||||||
NoSuchAlgorithmException e) {
|
NoSuchAlgorithmException e) {
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log.debug(String.format("%s has been initialized", this.getClass().getName()));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -110,12 +106,33 @@ public class GpasPseudonymGenerator implements Generator {
|
|||||||
|
|
||||||
@NotNull
|
@NotNull
|
||||||
public static String unwrapPseudonym(Parameters gPasPseudonymResult) {
|
public static String unwrapPseudonym(Parameters gPasPseudonymResult) {
|
||||||
Identifier pseudonym = (Identifier) gPasPseudonymResult.getParameter().stream().findFirst()
|
final var parameters = gPasPseudonymResult.getParameter().stream().findFirst();
|
||||||
.get().getPart().stream().filter(a -> a.getName().equals("pseudonym")).findFirst()
|
|
||||||
|
if (parameters.isEmpty()) {
|
||||||
|
throw new PseudonymRequestFailed("Empty HL7 parameters, cannot find first one");
|
||||||
|
}
|
||||||
|
|
||||||
|
final var identifier = (Identifier) parameters.get().getPart().stream()
|
||||||
|
.filter(a -> a.getName().equals("pseudonym"))
|
||||||
|
.findFirst()
|
||||||
.orElseGet(ParametersParameterComponent::new).getValue();
|
.orElseGet(ParametersParameterComponent::new).getValue();
|
||||||
|
|
||||||
// pseudonym
|
// pseudonym
|
||||||
return pseudonym.getSystem() + "|" + pseudonym.getValue();
|
return sanitizeValue(identifier.getValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Allow only filename friendly values
|
||||||
|
*
|
||||||
|
* @param psnValue GAPS pseudonym value
|
||||||
|
* @return cleaned up value
|
||||||
|
*/
|
||||||
|
public static String sanitizeValue(String psnValue) {
|
||||||
|
// pattern to match forbidden characters
|
||||||
|
String forbiddenCharsRegex = "[\\\\/:*?\"<>|;]";
|
||||||
|
|
||||||
|
// Replace all forbidden characters with underscores
|
||||||
|
return psnValue.replaceAll(forbiddenCharsRegex, "_");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -124,7 +141,6 @@ public class GpasPseudonymGenerator implements Generator {
|
|||||||
|
|
||||||
HttpEntity<String> requestEntity = new HttpEntity<>(gPasRequestBody, this.httpHeader);
|
HttpEntity<String> requestEntity = new HttpEntity<>(gPasRequestBody, this.httpHeader);
|
||||||
ResponseEntity<String> responseEntity;
|
ResponseEntity<String> responseEntity;
|
||||||
var restTemplate = getRestTemplete();
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
responseEntity = retryTemplate.execute(
|
responseEntity = retryTemplate.execute(
|
||||||
@ -176,31 +192,6 @@ public class GpasPseudonymGenerator implements Generator {
|
|||||||
return headers;
|
return headers;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected RetryTemplate defaultTemplate() {
|
|
||||||
RetryTemplate retryTemplate = new RetryTemplate();
|
|
||||||
ExponentialBackOffPolicy backOffPolicy = new ExponentialBackOffPolicy();
|
|
||||||
backOffPolicy.setInitialInterval(1000);
|
|
||||||
backOffPolicy.setMultiplier(1.25);
|
|
||||||
retryTemplate.setBackOffPolicy(backOffPolicy);
|
|
||||||
HashMap<Class<? extends Throwable>, Boolean> retryableExceptions = new HashMap<>();
|
|
||||||
retryableExceptions.put(RestClientException.class, true);
|
|
||||||
retryableExceptions.put(ConnectException.class, true);
|
|
||||||
RetryPolicy retryPolicy = new SimpleRetryPolicy(3, retryableExceptions);
|
|
||||||
retryTemplate.setRetryPolicy(retryPolicy);
|
|
||||||
|
|
||||||
retryTemplate.registerListener(new RetryListener() {
|
|
||||||
@Override
|
|
||||||
public <T, E extends Throwable> void onError(RetryContext context,
|
|
||||||
RetryCallback<T, E> callback, Throwable throwable) {
|
|
||||||
log.warn("HTTP Error occurred: {}. Retrying {}", throwable.getMessage(),
|
|
||||||
context.getRetryCount());
|
|
||||||
RetryListener.super.onError(context, callback, throwable);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
return retryTemplate;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Read SSL root certificate and return SSLContext
|
* Read SSL root certificate and return SSLContext
|
||||||
*
|
*
|
||||||
@ -236,14 +227,8 @@ public class GpasPseudonymGenerator implements Generator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected RestTemplate getRestTemplete() {
|
protected RestTemplate getRestTemplete() {
|
||||||
|
|
||||||
if (restTemplate != null) {
|
|
||||||
return restTemplate;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (customSslContext == null) {
|
if (customSslContext == null) {
|
||||||
restTemplate = new RestTemplate();
|
return new RestTemplate();
|
||||||
return restTemplate;
|
|
||||||
}
|
}
|
||||||
final var sslsf = new SSLConnectionSocketFactory(customSslContext);
|
final var sslsf = new SSLConnectionSocketFactory(customSslContext);
|
||||||
final Registry<ConnectionSocketFactory> socketFactoryRegistry = RegistryBuilder.<ConnectionSocketFactory>create()
|
final Registry<ConnectionSocketFactory> socketFactoryRegistry = RegistryBuilder.<ConnectionSocketFactory>create()
|
||||||
@ -256,7 +241,6 @@ public class GpasPseudonymGenerator implements Generator {
|
|||||||
|
|
||||||
final HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory(
|
final HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory(
|
||||||
httpClient);
|
httpClient);
|
||||||
restTemplate = new RestTemplate(requestFactory);
|
return new RestTemplate(requestFactory);
|
||||||
return restTemplate;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -20,9 +20,10 @@
|
|||||||
package dev.dnpm.etl.processor
|
package dev.dnpm.etl.processor
|
||||||
|
|
||||||
import org.springframework.boot.autoconfigure.SpringBootApplication
|
import org.springframework.boot.autoconfigure.SpringBootApplication
|
||||||
|
import org.springframework.boot.autoconfigure.security.servlet.SecurityAutoConfiguration
|
||||||
import org.springframework.boot.runApplication
|
import org.springframework.boot.runApplication
|
||||||
|
|
||||||
@SpringBootApplication
|
@SpringBootApplication(exclude = [SecurityAutoConfiguration::class])
|
||||||
class EtlProcessorApplication
|
class EtlProcessorApplication
|
||||||
|
|
||||||
fun main(args: Array<String>) {
|
fun main(args: Array<String>) {
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -19,12 +19,21 @@
|
|||||||
|
|
||||||
package dev.dnpm.etl.processor.config
|
package dev.dnpm.etl.processor.config
|
||||||
|
|
||||||
|
import dev.dnpm.etl.processor.security.Role
|
||||||
import org.springframework.boot.context.properties.ConfigurationProperties
|
import org.springframework.boot.context.properties.ConfigurationProperties
|
||||||
|
import org.springframework.boot.context.properties.DeprecatedConfigurationProperty
|
||||||
|
|
||||||
@ConfigurationProperties(AppConfigProperties.NAME)
|
@ConfigurationProperties(AppConfigProperties.NAME)
|
||||||
data class AppConfigProperties(
|
data class AppConfigProperties(
|
||||||
var bwhc_uri: String?,
|
var bwhcUri: String?,
|
||||||
var generator: PseudonymGenerator = PseudonymGenerator.BUILDIN
|
@get:DeprecatedConfigurationProperty(
|
||||||
|
reason = "Deprecated in favor of 'app.pseudonymize.generator'",
|
||||||
|
replacement = "app.pseudonymize.generator"
|
||||||
|
)
|
||||||
|
var pseudonymizer: PseudonymGenerator = PseudonymGenerator.BUILDIN,
|
||||||
|
var transformations: List<TransformationProperties> = listOf(),
|
||||||
|
var maxRetryAttempts: Int = 3,
|
||||||
|
var duplicationDetection: Boolean = true
|
||||||
) {
|
) {
|
||||||
companion object {
|
companion object {
|
||||||
const val NAME = "app"
|
const val NAME = "app"
|
||||||
@ -33,6 +42,7 @@ data class AppConfigProperties(
|
|||||||
|
|
||||||
@ConfigurationProperties(PseudonymizeConfigProperties.NAME)
|
@ConfigurationProperties(PseudonymizeConfigProperties.NAME)
|
||||||
data class PseudonymizeConfigProperties(
|
data class PseudonymizeConfigProperties(
|
||||||
|
var generator: PseudonymGenerator = PseudonymGenerator.BUILDIN,
|
||||||
val prefix: String = "UNKNOWN",
|
val prefix: String = "UNKNOWN",
|
||||||
) {
|
) {
|
||||||
companion object {
|
companion object {
|
||||||
@ -46,9 +56,11 @@ data class GPasConfigProperties(
|
|||||||
val target: String = "etl-processor",
|
val target: String = "etl-processor",
|
||||||
val username: String?,
|
val username: String?,
|
||||||
val password: String?,
|
val password: String?,
|
||||||
val sslCaLocation: String?,
|
@get:DeprecatedConfigurationProperty(
|
||||||
|
reason = "Deprecated in favor of including Root CA"
|
||||||
) {
|
)
|
||||||
|
val sslCaLocation: String?
|
||||||
|
) {
|
||||||
companion object {
|
companion object {
|
||||||
const val NAME = "app.pseudonymize.gpas"
|
const val NAME = "app.pseudonymize.gpas"
|
||||||
}
|
}
|
||||||
@ -63,10 +75,21 @@ data class RestTargetProperties(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ConfigurationProperties(KafkaTargetProperties.NAME)
|
@ConfigurationProperties(KafkaProperties.NAME)
|
||||||
data class KafkaTargetProperties(
|
data class KafkaProperties(
|
||||||
val topic: String = "etl-processor",
|
val inputTopic: String?,
|
||||||
val responseTopic: String = "${topic}_response",
|
val outputTopic: String = "etl-processor",
|
||||||
|
@get:DeprecatedConfigurationProperty(
|
||||||
|
reason = "Deprecated",
|
||||||
|
replacement = "outputTopic"
|
||||||
|
)
|
||||||
|
val topic: String = outputTopic,
|
||||||
|
val outputResponseTopic: String = "${outputTopic}_response",
|
||||||
|
@get:DeprecatedConfigurationProperty(
|
||||||
|
reason = "Deprecated",
|
||||||
|
replacement = "outputResponseTopic"
|
||||||
|
)
|
||||||
|
val responseTopic: String = outputResponseTopic,
|
||||||
val groupId: String = "${topic}_group",
|
val groupId: String = "${topic}_group",
|
||||||
val servers: String = ""
|
val servers: String = ""
|
||||||
) {
|
) {
|
||||||
@ -75,7 +98,26 @@ data class KafkaTargetProperties(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ConfigurationProperties(SecurityConfigProperties.NAME)
|
||||||
|
data class SecurityConfigProperties(
|
||||||
|
val adminUser: String?,
|
||||||
|
val adminPassword: String?,
|
||||||
|
val enableTokens: Boolean = false,
|
||||||
|
val enableOidc: Boolean = false,
|
||||||
|
val defaultNewUserRole: Role = Role.USER
|
||||||
|
) {
|
||||||
|
companion object {
|
||||||
|
const val NAME = "app.security"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
enum class PseudonymGenerator {
|
enum class PseudonymGenerator {
|
||||||
BUILDIN,
|
BUILDIN,
|
||||||
GPAS
|
GPAS
|
||||||
}
|
}
|
||||||
|
|
||||||
|
data class TransformationProperties(
|
||||||
|
val path: String,
|
||||||
|
val from: String,
|
||||||
|
val to: String
|
||||||
|
)
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -20,16 +20,35 @@
|
|||||||
package dev.dnpm.etl.processor.config
|
package dev.dnpm.etl.processor.config
|
||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
import dev.dnpm.etl.processor.monitoring.ReportService
|
import dev.dnpm.etl.processor.monitoring.*
|
||||||
import dev.dnpm.etl.processor.pseudonym.AnonymizingGenerator
|
import dev.dnpm.etl.processor.pseudonym.AnonymizingGenerator
|
||||||
import dev.dnpm.etl.processor.pseudonym.Generator
|
import dev.dnpm.etl.processor.pseudonym.Generator
|
||||||
import dev.dnpm.etl.processor.pseudonym.GpasPseudonymGenerator
|
import dev.dnpm.etl.processor.pseudonym.GpasPseudonymGenerator
|
||||||
import dev.dnpm.etl.processor.pseudonym.PseudonymizeService
|
import dev.dnpm.etl.processor.pseudonym.PseudonymizeService
|
||||||
|
import dev.dnpm.etl.processor.services.TokenRepository
|
||||||
|
import dev.dnpm.etl.processor.services.TokenService
|
||||||
|
import dev.dnpm.etl.processor.services.Transformation
|
||||||
|
import dev.dnpm.etl.processor.services.TransformationService
|
||||||
|
import org.slf4j.LoggerFactory
|
||||||
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty
|
||||||
import org.springframework.boot.context.properties.EnableConfigurationProperties
|
import org.springframework.boot.context.properties.EnableConfigurationProperties
|
||||||
import org.springframework.context.annotation.Bean
|
import org.springframework.context.annotation.Bean
|
||||||
import org.springframework.context.annotation.Configuration
|
import org.springframework.context.annotation.Configuration
|
||||||
|
import org.springframework.retry.RetryCallback
|
||||||
|
import org.springframework.retry.RetryContext
|
||||||
|
import org.springframework.retry.RetryListener
|
||||||
|
import org.springframework.retry.policy.SimpleRetryPolicy
|
||||||
|
import org.springframework.retry.support.RetryTemplate
|
||||||
|
import org.springframework.retry.support.RetryTemplateBuilder
|
||||||
|
import org.springframework.scheduling.annotation.EnableScheduling
|
||||||
|
import org.springframework.security.crypto.password.PasswordEncoder
|
||||||
|
import org.springframework.security.provisioning.InMemoryUserDetailsManager
|
||||||
|
import org.springframework.web.client.RestTemplate
|
||||||
import reactor.core.publisher.Sinks
|
import reactor.core.publisher.Sinks
|
||||||
|
import kotlin.time.Duration.Companion.seconds
|
||||||
|
import kotlin.time.toJavaDuration
|
||||||
|
|
||||||
|
|
||||||
@Configuration
|
@Configuration
|
||||||
@EnableConfigurationProperties(
|
@EnableConfigurationProperties(
|
||||||
@ -39,20 +58,42 @@ import reactor.core.publisher.Sinks
|
|||||||
GPasConfigProperties::class
|
GPasConfigProperties::class
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
@EnableScheduling
|
||||||
class AppConfiguration {
|
class AppConfiguration {
|
||||||
|
|
||||||
@ConditionalOnProperty(value = ["app.pseudonymizer"], havingValue = "GPAS")
|
private val logger = LoggerFactory.getLogger(AppConfiguration::class.java)
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
fun gpasPseudonymGenerator(configProperties: GPasConfigProperties): Generator {
|
fun restTemplate(): RestTemplate {
|
||||||
return GpasPseudonymGenerator(configProperties)
|
return RestTemplate()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ConditionalOnProperty(value = ["app.pseudonymizer"], havingValue = "BUILDIN", matchIfMissing = true)
|
@ConditionalOnProperty(value = ["app.pseudonymize.generator"], havingValue = "GPAS")
|
||||||
|
@Bean
|
||||||
|
fun gpasPseudonymGenerator(configProperties: GPasConfigProperties, retryTemplate: RetryTemplate): Generator {
|
||||||
|
return GpasPseudonymGenerator(configProperties, retryTemplate)
|
||||||
|
}
|
||||||
|
|
||||||
|
@ConditionalOnProperty(value = ["app.pseudonymize.generator"], havingValue = "BUILDIN", matchIfMissing = true)
|
||||||
@Bean
|
@Bean
|
||||||
fun buildinPseudonymGenerator(): Generator {
|
fun buildinPseudonymGenerator(): Generator {
|
||||||
return AnonymizingGenerator()
|
return AnonymizingGenerator()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ConditionalOnProperty(value = ["app.pseudonymizer"], havingValue = "GPAS")
|
||||||
|
@ConditionalOnMissingBean
|
||||||
|
@Bean
|
||||||
|
fun gpasPseudonymGeneratorOnDeprecatedProperty(configProperties: GPasConfigProperties, retryTemplate: RetryTemplate): Generator {
|
||||||
|
return GpasPseudonymGenerator(configProperties, retryTemplate)
|
||||||
|
}
|
||||||
|
|
||||||
|
@ConditionalOnProperty(value = ["app.pseudonymizer"], havingValue = "BUILDIN")
|
||||||
|
@ConditionalOnMissingBean
|
||||||
|
@Bean
|
||||||
|
fun buildinPseudonymGeneratorOnDeprecatedProperty(): Generator {
|
||||||
|
return AnonymizingGenerator()
|
||||||
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
fun pseudonymizeService(
|
fun pseudonymizeService(
|
||||||
generator: Generator,
|
generator: Generator,
|
||||||
@ -66,10 +107,71 @@ class AppConfiguration {
|
|||||||
return ReportService(objectMapper)
|
return ReportService(objectMapper)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
fun transformationService(
|
||||||
|
objectMapper: ObjectMapper,
|
||||||
|
configProperties: AppConfigProperties
|
||||||
|
): TransformationService {
|
||||||
|
logger.info("Apply ${configProperties.transformations.size} transformation rules")
|
||||||
|
return TransformationService(objectMapper, configProperties.transformations.map {
|
||||||
|
Transformation.of(it.path) from it.from to it.to
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
fun retryTemplate(configProperties: AppConfigProperties): RetryTemplate {
|
||||||
|
return RetryTemplateBuilder()
|
||||||
|
.notRetryOn(IllegalArgumentException::class.java)
|
||||||
|
.exponentialBackoff(2.seconds.toJavaDuration(), 1.25, 5.seconds.toJavaDuration())
|
||||||
|
.customPolicy(SimpleRetryPolicy(configProperties.maxRetryAttempts))
|
||||||
|
.withListener(object : RetryListener {
|
||||||
|
override fun <T : Any, E : Throwable> onError(
|
||||||
|
context: RetryContext,
|
||||||
|
callback: RetryCallback<T, E>,
|
||||||
|
throwable: Throwable
|
||||||
|
) {
|
||||||
|
logger.warn("Error occured: {}. Retrying {}", throwable.message, context.retryCount)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.build()
|
||||||
|
}
|
||||||
|
|
||||||
|
@ConditionalOnProperty(value = ["app.security.enable-tokens"], havingValue = "true")
|
||||||
|
@Bean
|
||||||
|
fun tokenService(userDetailsManager: InMemoryUserDetailsManager, passwordEncoder: PasswordEncoder, tokenRepository: TokenRepository): TokenService {
|
||||||
|
return TokenService(userDetailsManager, passwordEncoder, tokenRepository)
|
||||||
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
fun statisticsUpdateProducer(): Sinks.Many<Any> {
|
fun statisticsUpdateProducer(): Sinks.Many<Any> {
|
||||||
return Sinks.many().multicast().directBestEffort()
|
return Sinks.many().multicast().directBestEffort()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
fun connectionCheckUpdateProducer(): Sinks.Many<ConnectionCheckResult> {
|
||||||
|
return Sinks.many().multicast().onBackpressureBuffer()
|
||||||
|
}
|
||||||
|
|
||||||
|
@ConditionalOnProperty(value = ["app.pseudonymize.generator"], havingValue = "GPAS")
|
||||||
|
@Bean
|
||||||
|
fun gPasConnectionCheckService(
|
||||||
|
restTemplate: RestTemplate,
|
||||||
|
gPasConfigProperties: GPasConfigProperties,
|
||||||
|
connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>
|
||||||
|
): ConnectionCheckService {
|
||||||
|
return GPasConnectionCheckService(restTemplate, gPasConfigProperties, connectionCheckUpdateProducer)
|
||||||
|
}
|
||||||
|
|
||||||
|
@ConditionalOnProperty(value = ["app.pseudonymizer"], havingValue = "GPAS")
|
||||||
|
@ConditionalOnMissingBean
|
||||||
|
@Bean
|
||||||
|
fun gPasConnectionCheckServiceOnDeprecatedProperty(
|
||||||
|
restTemplate: RestTemplate,
|
||||||
|
gPasConfigProperties: GPasConfigProperties,
|
||||||
|
connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>
|
||||||
|
): ConnectionCheckService {
|
||||||
|
return GPasConnectionCheckService(restTemplate, gPasConfigProperties, connectionCheckUpdateProducer)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -20,8 +20,13 @@
|
|||||||
package dev.dnpm.etl.processor.config
|
package dev.dnpm.etl.processor.config
|
||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import dev.dnpm.etl.processor.input.KafkaInputListener
|
||||||
|
import dev.dnpm.etl.processor.monitoring.ConnectionCheckResult
|
||||||
|
import dev.dnpm.etl.processor.monitoring.ConnectionCheckService
|
||||||
|
import dev.dnpm.etl.processor.monitoring.KafkaConnectionCheckService
|
||||||
import dev.dnpm.etl.processor.output.KafkaMtbFileSender
|
import dev.dnpm.etl.processor.output.KafkaMtbFileSender
|
||||||
import dev.dnpm.etl.processor.output.MtbFileSender
|
import dev.dnpm.etl.processor.output.MtbFileSender
|
||||||
|
import dev.dnpm.etl.processor.services.RequestProcessor
|
||||||
import dev.dnpm.etl.processor.services.kafka.KafkaResponseProcessor
|
import dev.dnpm.etl.processor.services.kafka.KafkaResponseProcessor
|
||||||
import org.slf4j.LoggerFactory
|
import org.slf4j.LoggerFactory
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean
|
||||||
@ -35,12 +40,14 @@ import org.springframework.kafka.core.ConsumerFactory
|
|||||||
import org.springframework.kafka.core.KafkaTemplate
|
import org.springframework.kafka.core.KafkaTemplate
|
||||||
import org.springframework.kafka.listener.ContainerProperties
|
import org.springframework.kafka.listener.ContainerProperties
|
||||||
import org.springframework.kafka.listener.KafkaMessageListenerContainer
|
import org.springframework.kafka.listener.KafkaMessageListenerContainer
|
||||||
|
import org.springframework.retry.support.RetryTemplate
|
||||||
|
import reactor.core.publisher.Sinks
|
||||||
|
|
||||||
@Configuration
|
@Configuration
|
||||||
@EnableConfigurationProperties(
|
@EnableConfigurationProperties(
|
||||||
value = [KafkaTargetProperties::class]
|
value = [KafkaProperties::class]
|
||||||
)
|
)
|
||||||
@ConditionalOnProperty(value = ["app.kafka.topic", "app.kafka.servers"])
|
@ConditionalOnProperty(value = ["app.kafka.servers"])
|
||||||
@ConditionalOnMissingBean(MtbFileSender::class)
|
@ConditionalOnMissingBean(MtbFileSender::class)
|
||||||
@Order(-5)
|
@Order(-5)
|
||||||
class AppKafkaConfiguration {
|
class AppKafkaConfiguration {
|
||||||
@ -50,20 +57,21 @@ class AppKafkaConfiguration {
|
|||||||
@Bean
|
@Bean
|
||||||
fun kafkaMtbFileSender(
|
fun kafkaMtbFileSender(
|
||||||
kafkaTemplate: KafkaTemplate<String, String>,
|
kafkaTemplate: KafkaTemplate<String, String>,
|
||||||
kafkaTargetProperties: KafkaTargetProperties,
|
kafkaProperties: KafkaProperties,
|
||||||
|
retryTemplate: RetryTemplate,
|
||||||
objectMapper: ObjectMapper
|
objectMapper: ObjectMapper
|
||||||
): MtbFileSender {
|
): MtbFileSender {
|
||||||
logger.info("Selected 'KafkaMtbFileSender'")
|
logger.info("Selected 'KafkaMtbFileSender'")
|
||||||
return KafkaMtbFileSender(kafkaTemplate, kafkaTargetProperties, objectMapper)
|
return KafkaMtbFileSender(kafkaTemplate, kafkaProperties, retryTemplate, objectMapper)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
fun kafkaListenerContainer(
|
fun kafkaResponseListenerContainer(
|
||||||
consumerFactory: ConsumerFactory<String, String>,
|
consumerFactory: ConsumerFactory<String, String>,
|
||||||
kafkaTargetProperties: KafkaTargetProperties,
|
kafkaProperties: KafkaProperties,
|
||||||
kafkaResponseProcessor: KafkaResponseProcessor
|
kafkaResponseProcessor: KafkaResponseProcessor
|
||||||
): KafkaMessageListenerContainer<String, String> {
|
): KafkaMessageListenerContainer<String, String> {
|
||||||
val containerProperties = ContainerProperties(kafkaTargetProperties.responseTopic)
|
val containerProperties = ContainerProperties(kafkaProperties.responseTopic)
|
||||||
containerProperties.messageListener = kafkaResponseProcessor
|
containerProperties.messageListener = kafkaResponseProcessor
|
||||||
return KafkaMessageListenerContainer(consumerFactory, containerProperties)
|
return KafkaMessageListenerContainer(consumerFactory, containerProperties)
|
||||||
}
|
}
|
||||||
@ -76,4 +84,33 @@ class AppKafkaConfiguration {
|
|||||||
return KafkaResponseProcessor(applicationEventPublisher, objectMapper)
|
return KafkaResponseProcessor(applicationEventPublisher, objectMapper)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnProperty(value = ["app.kafka.input-topic"])
|
||||||
|
fun kafkaInputListenerContainer(
|
||||||
|
consumerFactory: ConsumerFactory<String, String>,
|
||||||
|
kafkaProperties: KafkaProperties,
|
||||||
|
kafkaInputListener: KafkaInputListener
|
||||||
|
): KafkaMessageListenerContainer<String, String> {
|
||||||
|
val containerProperties = ContainerProperties(kafkaProperties.inputTopic)
|
||||||
|
containerProperties.messageListener = kafkaInputListener
|
||||||
|
return KafkaMessageListenerContainer(consumerFactory, containerProperties)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnProperty(value = ["app.kafka.input-topic"])
|
||||||
|
fun kafkaInputListener(
|
||||||
|
requestProcessor: RequestProcessor,
|
||||||
|
objectMapper: ObjectMapper
|
||||||
|
): KafkaInputListener {
|
||||||
|
return KafkaInputListener(requestProcessor, objectMapper)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
fun kafkaConnectionCheckService(
|
||||||
|
consumerFactory: ConsumerFactory<String, String>,
|
||||||
|
connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>
|
||||||
|
): ConnectionCheckService {
|
||||||
|
return KafkaConnectionCheckService(consumerFactory.createConsumer(), connectionCheckUpdateProducer)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -19,6 +19,9 @@
|
|||||||
|
|
||||||
package dev.dnpm.etl.processor.config
|
package dev.dnpm.etl.processor.config
|
||||||
|
|
||||||
|
import dev.dnpm.etl.processor.monitoring.ConnectionCheckResult
|
||||||
|
import dev.dnpm.etl.processor.monitoring.ConnectionCheckService
|
||||||
|
import dev.dnpm.etl.processor.monitoring.RestConnectionCheckService
|
||||||
import dev.dnpm.etl.processor.output.MtbFileSender
|
import dev.dnpm.etl.processor.output.MtbFileSender
|
||||||
import dev.dnpm.etl.processor.output.RestMtbFileSender
|
import dev.dnpm.etl.processor.output.RestMtbFileSender
|
||||||
import org.slf4j.LoggerFactory
|
import org.slf4j.LoggerFactory
|
||||||
@ -28,7 +31,9 @@ import org.springframework.boot.context.properties.EnableConfigurationProperties
|
|||||||
import org.springframework.context.annotation.Bean
|
import org.springframework.context.annotation.Bean
|
||||||
import org.springframework.context.annotation.Configuration
|
import org.springframework.context.annotation.Configuration
|
||||||
import org.springframework.core.annotation.Order
|
import org.springframework.core.annotation.Order
|
||||||
|
import org.springframework.retry.support.RetryTemplate
|
||||||
import org.springframework.web.client.RestTemplate
|
import org.springframework.web.client.RestTemplate
|
||||||
|
import reactor.core.publisher.Sinks
|
||||||
|
|
||||||
@Configuration
|
@Configuration
|
||||||
@EnableConfigurationProperties(
|
@EnableConfigurationProperties(
|
||||||
@ -44,14 +49,22 @@ class AppRestConfiguration {
|
|||||||
private val logger = LoggerFactory.getLogger(AppRestConfiguration::class.java)
|
private val logger = LoggerFactory.getLogger(AppRestConfiguration::class.java)
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
fun restTemplate(): RestTemplate {
|
fun restMtbFileSender(
|
||||||
return RestTemplate()
|
restTemplate: RestTemplate,
|
||||||
|
restTargetProperties: RestTargetProperties,
|
||||||
|
retryTemplate: RetryTemplate
|
||||||
|
): MtbFileSender {
|
||||||
|
logger.info("Selected 'RestMtbFileSender'")
|
||||||
|
return RestMtbFileSender(restTemplate, restTargetProperties, retryTemplate)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
fun restMtbFileSender(restTemplate: RestTemplate, restTargetProperties: RestTargetProperties): MtbFileSender {
|
fun restConnectionCheckService(
|
||||||
logger.info("Selected 'RestMtbFileSender'")
|
restTemplate: RestTemplate,
|
||||||
return RestMtbFileSender(restTemplate, restTargetProperties)
|
restTargetProperties: RestTargetProperties,
|
||||||
|
connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>
|
||||||
|
): ConnectionCheckService {
|
||||||
|
return RestConnectionCheckService(restTemplate, restTargetProperties, connectionCheckUpdateProducer)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,180 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.config
|
||||||
|
|
||||||
|
import dev.dnpm.etl.processor.security.UserRole
|
||||||
|
import dev.dnpm.etl.processor.security.UserRoleRepository
|
||||||
|
import dev.dnpm.etl.processor.services.UserRoleService
|
||||||
|
import org.slf4j.LoggerFactory
|
||||||
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty
|
||||||
|
import org.springframework.boot.context.properties.EnableConfigurationProperties
|
||||||
|
import org.springframework.context.annotation.Bean
|
||||||
|
import org.springframework.context.annotation.Configuration
|
||||||
|
import org.springframework.security.config.annotation.web.builders.HttpSecurity
|
||||||
|
import org.springframework.security.config.annotation.web.configuration.EnableWebSecurity
|
||||||
|
import org.springframework.security.config.annotation.web.invoke
|
||||||
|
import org.springframework.security.core.authority.SimpleGrantedAuthority
|
||||||
|
import org.springframework.security.core.authority.mapping.GrantedAuthoritiesMapper
|
||||||
|
import org.springframework.security.core.session.SessionRegistry
|
||||||
|
import org.springframework.security.core.session.SessionRegistryImpl
|
||||||
|
import org.springframework.security.core.userdetails.User
|
||||||
|
import org.springframework.security.core.userdetails.UserDetails
|
||||||
|
import org.springframework.security.crypto.factory.PasswordEncoderFactories
|
||||||
|
import org.springframework.security.crypto.password.PasswordEncoder
|
||||||
|
import org.springframework.security.oauth2.core.oidc.user.OidcUserAuthority
|
||||||
|
import org.springframework.security.provisioning.InMemoryUserDetailsManager
|
||||||
|
import org.springframework.security.web.SecurityFilterChain
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
@EnableConfigurationProperties(
|
||||||
|
value = [
|
||||||
|
SecurityConfigProperties::class
|
||||||
|
]
|
||||||
|
)
|
||||||
|
@ConditionalOnProperty(value = ["app.security.admin-user"])
|
||||||
|
@EnableWebSecurity
|
||||||
|
class AppSecurityConfiguration(
|
||||||
|
private val securityConfigProperties: SecurityConfigProperties
|
||||||
|
) {
|
||||||
|
|
||||||
|
private val logger = LoggerFactory.getLogger(AppSecurityConfiguration::class.java)
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
fun userDetailsService(passwordEncoder: PasswordEncoder): InMemoryUserDetailsManager {
|
||||||
|
val adminUser = if (securityConfigProperties.adminUser.isNullOrBlank()) {
|
||||||
|
logger.warn("Using random Admin User: admin")
|
||||||
|
"admin"
|
||||||
|
} else {
|
||||||
|
securityConfigProperties.adminUser
|
||||||
|
}
|
||||||
|
|
||||||
|
val adminPassword = if (securityConfigProperties.adminPassword.isNullOrBlank()) {
|
||||||
|
val random = UUID.randomUUID().toString()
|
||||||
|
logger.warn("Using random Admin Passwort: {}", random)
|
||||||
|
passwordEncoder.encode(random)
|
||||||
|
} else {
|
||||||
|
securityConfigProperties.adminPassword
|
||||||
|
}
|
||||||
|
|
||||||
|
val user: UserDetails = User.withUsername(adminUser)
|
||||||
|
.password(adminPassword)
|
||||||
|
.roles("ADMIN")
|
||||||
|
.build()
|
||||||
|
|
||||||
|
return InMemoryUserDetailsManager(user)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnProperty(value = ["app.security.enable-oidc"], havingValue = "true")
|
||||||
|
fun filterChainOidc(http: HttpSecurity, passwordEncoder: PasswordEncoder, userRoleRepository: UserRoleRepository, sessionRegistry: SessionRegistry): SecurityFilterChain {
|
||||||
|
http {
|
||||||
|
authorizeRequests {
|
||||||
|
authorize("/configs/**", hasRole("ADMIN"))
|
||||||
|
authorize("/mtbfile/**", hasAnyRole("MTBFILE"))
|
||||||
|
authorize("/report/**", hasAnyRole("ADMIN", "USER"))
|
||||||
|
authorize("*.css", permitAll)
|
||||||
|
authorize("*.ico", permitAll)
|
||||||
|
authorize("*.jpeg", permitAll)
|
||||||
|
authorize("*.js", permitAll)
|
||||||
|
authorize("*.svg", permitAll)
|
||||||
|
authorize("*.css", permitAll)
|
||||||
|
authorize("/login/**", permitAll)
|
||||||
|
authorize(anyRequest, permitAll)
|
||||||
|
}
|
||||||
|
httpBasic {
|
||||||
|
realmName = "ETL-Processor"
|
||||||
|
}
|
||||||
|
formLogin {
|
||||||
|
loginPage = "/login"
|
||||||
|
}
|
||||||
|
oauth2Login {
|
||||||
|
loginPage = "/login"
|
||||||
|
}
|
||||||
|
sessionManagement {
|
||||||
|
sessionConcurrency {
|
||||||
|
maximumSessions = 1
|
||||||
|
expiredUrl = "/login?expired"
|
||||||
|
}
|
||||||
|
sessionFixation {
|
||||||
|
newSession()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
csrf { disable() }
|
||||||
|
}
|
||||||
|
return http.build()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnProperty(value = ["app.security.enable-oidc"], havingValue = "true")
|
||||||
|
fun grantedAuthoritiesMapper(userRoleRepository: UserRoleRepository, appSecurityConfigProperties: SecurityConfigProperties): GrantedAuthoritiesMapper {
|
||||||
|
return GrantedAuthoritiesMapper { grantedAuthority ->
|
||||||
|
grantedAuthority.filterIsInstance<OidcUserAuthority>()
|
||||||
|
.onEach {
|
||||||
|
val userRole = userRoleRepository.findByUsername(it.userInfo.preferredUsername)
|
||||||
|
if (userRole.isEmpty) {
|
||||||
|
userRoleRepository.save(UserRole(null, it.userInfo.preferredUsername, appSecurityConfigProperties.defaultNewUserRole))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.map {
|
||||||
|
val userRole = userRoleRepository.findByUsername(it.userInfo.preferredUsername)
|
||||||
|
SimpleGrantedAuthority("ROLE_${userRole.get().role.toString().uppercase()}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnProperty(value = ["app.security.enable-oidc"], havingValue = "false", matchIfMissing = true)
|
||||||
|
fun filterChain(http: HttpSecurity, passwordEncoder: PasswordEncoder): SecurityFilterChain {
|
||||||
|
http {
|
||||||
|
authorizeRequests {
|
||||||
|
authorize("/configs/**", hasRole("ADMIN"))
|
||||||
|
authorize("/mtbfile/**", hasAnyRole("MTBFILE"))
|
||||||
|
authorize("/report/**", hasRole("ADMIN"))
|
||||||
|
authorize(anyRequest, permitAll)
|
||||||
|
}
|
||||||
|
httpBasic {
|
||||||
|
realmName = "ETL-Processor"
|
||||||
|
}
|
||||||
|
formLogin {
|
||||||
|
loginPage = "/login"
|
||||||
|
}
|
||||||
|
csrf { disable() }
|
||||||
|
}
|
||||||
|
return http.build()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
fun sessionRegistry(): SessionRegistry {
|
||||||
|
return SessionRegistryImpl()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
fun passwordEncoder(): PasswordEncoder {
|
||||||
|
return PasswordEncoderFactories.createDelegatingPasswordEncoder()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
@ConditionalOnProperty(value = ["app.security.enable-oidc"], havingValue = "true")
|
||||||
|
fun userRoleService(userRoleRepository: UserRoleRepository, sessionRegistry: SessionRegistry): UserRoleService {
|
||||||
|
return UserRoleService(userRoleRepository, sessionRegistry)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,61 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.input
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import de.ukw.ccc.bwhc.dto.Consent
|
||||||
|
import de.ukw.ccc.bwhc.dto.MtbFile
|
||||||
|
import dev.dnpm.etl.processor.services.RequestProcessor
|
||||||
|
import org.apache.kafka.clients.consumer.ConsumerRecord
|
||||||
|
import org.slf4j.LoggerFactory
|
||||||
|
import org.springframework.kafka.listener.MessageListener
|
||||||
|
|
||||||
|
class KafkaInputListener(
|
||||||
|
private val requestProcessor: RequestProcessor,
|
||||||
|
private val objectMapper: ObjectMapper
|
||||||
|
) : MessageListener<String, String> {
|
||||||
|
private val logger = LoggerFactory.getLogger(KafkaInputListener::class.java)
|
||||||
|
|
||||||
|
override fun onMessage(data: ConsumerRecord<String, String>) {
|
||||||
|
val mtbFile = objectMapper.readValue(data.value(), MtbFile::class.java)
|
||||||
|
val firstRequestIdHeader = data.headers().headers("requestId")?.firstOrNull()
|
||||||
|
val requestId = if (null != firstRequestIdHeader) {
|
||||||
|
String(firstRequestIdHeader.value())
|
||||||
|
} else {
|
||||||
|
""
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mtbFile.consent.status == Consent.Status.ACTIVE) {
|
||||||
|
logger.debug("Accepted MTB File for processing")
|
||||||
|
if (requestId.isBlank()) {
|
||||||
|
requestProcessor.processMtbFile(mtbFile)
|
||||||
|
} else {
|
||||||
|
requestProcessor.processMtbFile(mtbFile, requestId)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
logger.debug("Accepted MTB File and process deletion")
|
||||||
|
if (requestId.isBlank()) {
|
||||||
|
requestProcessor.processDeletion(mtbFile.patient.id)
|
||||||
|
} else {
|
||||||
|
requestProcessor.processDeletion(mtbFile.patient.id, requestId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -17,7 +17,7 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package dev.dnpm.etl.processor.web
|
package dev.dnpm.etl.processor.input
|
||||||
|
|
||||||
import de.ukw.ccc.bwhc.dto.Consent
|
import de.ukw.ccc.bwhc.dto.Consent
|
||||||
import de.ukw.ccc.bwhc.dto.MtbFile
|
import de.ukw.ccc.bwhc.dto.MtbFile
|
||||||
@ -27,13 +27,19 @@ import org.springframework.http.ResponseEntity
|
|||||||
import org.springframework.web.bind.annotation.*
|
import org.springframework.web.bind.annotation.*
|
||||||
|
|
||||||
@RestController
|
@RestController
|
||||||
|
@RequestMapping(path = ["mtbfile"])
|
||||||
class MtbFileRestController(
|
class MtbFileRestController(
|
||||||
private val requestProcessor: RequestProcessor,
|
private val requestProcessor: RequestProcessor,
|
||||||
) {
|
) {
|
||||||
|
|
||||||
private val logger = LoggerFactory.getLogger(MtbFileRestController::class.java)
|
private val logger = LoggerFactory.getLogger(MtbFileRestController::class.java)
|
||||||
|
|
||||||
@PostMapping(path = ["/mtbfile"])
|
@GetMapping
|
||||||
|
fun info(): ResponseEntity<String> {
|
||||||
|
return ResponseEntity.ok("Test")
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping
|
||||||
fun mtbFile(@RequestBody mtbFile: MtbFile): ResponseEntity<Void> {
|
fun mtbFile(@RequestBody mtbFile: MtbFile): ResponseEntity<Void> {
|
||||||
if (mtbFile.consent.status == Consent.Status.ACTIVE) {
|
if (mtbFile.consent.status == Consent.Status.ACTIVE) {
|
||||||
logger.debug("Accepted MTB File for processing")
|
logger.debug("Accepted MTB File for processing")
|
||||||
@ -45,7 +51,7 @@ class MtbFileRestController(
|
|||||||
return ResponseEntity.accepted().build()
|
return ResponseEntity.accepted().build()
|
||||||
}
|
}
|
||||||
|
|
||||||
@DeleteMapping(path = ["/mtbfile/{patientId}"])
|
@DeleteMapping(path = ["{patientId}"])
|
||||||
fun deleteData(@PathVariable patientId: String): ResponseEntity<Void> {
|
fun deleteData(@PathVariable patientId: String): ResponseEntity<Void> {
|
||||||
logger.debug("Accepted patient ID to process deletion")
|
logger.debug("Accepted patient ID to process deletion")
|
||||||
requestProcessor.processDeletion(patientId)
|
requestProcessor.processDeletion(patientId)
|
@ -0,0 +1,162 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.monitoring
|
||||||
|
|
||||||
|
import dev.dnpm.etl.processor.config.GPasConfigProperties
|
||||||
|
import dev.dnpm.etl.processor.config.RestTargetProperties
|
||||||
|
import jakarta.annotation.PostConstruct
|
||||||
|
import org.apache.kafka.clients.consumer.Consumer
|
||||||
|
import org.apache.kafka.common.errors.TimeoutException
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier
|
||||||
|
import org.springframework.http.HttpEntity
|
||||||
|
import org.springframework.http.HttpHeaders
|
||||||
|
import org.springframework.http.HttpMethod
|
||||||
|
import org.springframework.http.HttpStatus
|
||||||
|
import org.springframework.http.MediaType
|
||||||
|
import org.springframework.http.RequestEntity
|
||||||
|
import org.springframework.scheduling.annotation.Scheduled
|
||||||
|
import org.springframework.web.client.RestTemplate
|
||||||
|
import org.springframework.web.util.UriComponentsBuilder
|
||||||
|
import reactor.core.publisher.Sinks
|
||||||
|
import kotlin.time.Duration.Companion.seconds
|
||||||
|
import kotlin.time.toJavaDuration
|
||||||
|
|
||||||
|
interface ConnectionCheckService {
|
||||||
|
|
||||||
|
fun connectionAvailable(): Boolean
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
interface OutputConnectionCheckService : ConnectionCheckService
|
||||||
|
|
||||||
|
sealed class ConnectionCheckResult {
|
||||||
|
|
||||||
|
abstract val available: Boolean
|
||||||
|
|
||||||
|
data class KafkaConnectionCheckResult(override val available: Boolean) : ConnectionCheckResult()
|
||||||
|
data class RestConnectionCheckResult(override val available: Boolean) : ConnectionCheckResult()
|
||||||
|
data class GPasConnectionCheckResult(override val available: Boolean) : ConnectionCheckResult()
|
||||||
|
}
|
||||||
|
|
||||||
|
class KafkaConnectionCheckService(
|
||||||
|
private val consumer: Consumer<String, String>,
|
||||||
|
@Qualifier("connectionCheckUpdateProducer")
|
||||||
|
private val connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>
|
||||||
|
) : OutputConnectionCheckService {
|
||||||
|
|
||||||
|
private var connectionAvailable: Boolean = false
|
||||||
|
|
||||||
|
|
||||||
|
@PostConstruct
|
||||||
|
@Scheduled(cron = "0 * * * * *")
|
||||||
|
fun check() {
|
||||||
|
connectionAvailable = try {
|
||||||
|
null != consumer.listTopics(5.seconds.toJavaDuration())
|
||||||
|
} catch (e: TimeoutException) {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
connectionCheckUpdateProducer.emitNext(
|
||||||
|
ConnectionCheckResult.KafkaConnectionCheckResult(connectionAvailable),
|
||||||
|
Sinks.EmitFailureHandler.FAIL_FAST
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun connectionAvailable(): Boolean {
|
||||||
|
return this.connectionAvailable
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class RestConnectionCheckService(
|
||||||
|
private val restTemplate: RestTemplate,
|
||||||
|
private val restTargetProperties: RestTargetProperties,
|
||||||
|
@Qualifier("connectionCheckUpdateProducer")
|
||||||
|
private val connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>
|
||||||
|
) : OutputConnectionCheckService {
|
||||||
|
|
||||||
|
private var connectionAvailable: Boolean = false
|
||||||
|
|
||||||
|
@PostConstruct
|
||||||
|
@Scheduled(cron = "0 * * * * *")
|
||||||
|
fun check() {
|
||||||
|
connectionAvailable = try {
|
||||||
|
restTemplate.getForEntity(
|
||||||
|
restTargetProperties.uri?.replace("/etl/api", "").toString(),
|
||||||
|
String::class.java
|
||||||
|
).statusCode == HttpStatus.OK
|
||||||
|
} catch (e: Exception) {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
connectionCheckUpdateProducer.emitNext(
|
||||||
|
ConnectionCheckResult.RestConnectionCheckResult(connectionAvailable),
|
||||||
|
Sinks.EmitFailureHandler.FAIL_FAST
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun connectionAvailable(): Boolean {
|
||||||
|
return this.connectionAvailable
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class GPasConnectionCheckService(
|
||||||
|
private val restTemplate: RestTemplate,
|
||||||
|
private val gPasConfigProperties: GPasConfigProperties,
|
||||||
|
@Qualifier("connectionCheckUpdateProducer")
|
||||||
|
private val connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>
|
||||||
|
) : ConnectionCheckService {
|
||||||
|
|
||||||
|
private var connectionAvailable: Boolean = false
|
||||||
|
|
||||||
|
@PostConstruct
|
||||||
|
@Scheduled(cron = "0 * * * * *")
|
||||||
|
fun check() {
|
||||||
|
connectionAvailable = try {
|
||||||
|
val uri = UriComponentsBuilder.fromUriString(
|
||||||
|
gPasConfigProperties.uri?.replace("/\$pseudonymizeAllowCreate", "/\$pseudonymize").toString()
|
||||||
|
)
|
||||||
|
.queryParam("target", gPasConfigProperties.target)
|
||||||
|
.queryParam("original", "???")
|
||||||
|
.build().toUri()
|
||||||
|
|
||||||
|
val headers = HttpHeaders()
|
||||||
|
headers.contentType = MediaType.APPLICATION_JSON
|
||||||
|
if (!gPasConfigProperties.username.isNullOrBlank() && !gPasConfigProperties.password.isNullOrBlank()) {
|
||||||
|
headers.setBasicAuth(gPasConfigProperties.username, gPasConfigProperties.password)
|
||||||
|
}
|
||||||
|
restTemplate.exchange(
|
||||||
|
uri,
|
||||||
|
HttpMethod.GET,
|
||||||
|
HttpEntity<Void>(headers),
|
||||||
|
Void::class.java
|
||||||
|
).statusCode == HttpStatus.OK
|
||||||
|
} catch (e: Exception) {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
connectionCheckUpdateProducer.emitNext(
|
||||||
|
ConnectionCheckResult.GPasConnectionCheckResult(connectionAvailable),
|
||||||
|
Sinks.EmitFailureHandler.FAIL_FAST
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun connectionAvailable(): Boolean {
|
||||||
|
return this.connectionAvailable
|
||||||
|
}
|
||||||
|
}
|
@ -34,7 +34,10 @@ class ReportService(
|
|||||||
return listOf()
|
return listOf()
|
||||||
}
|
}
|
||||||
return try {
|
return try {
|
||||||
objectMapper.readValue(dataQualityReport, DataQualityReport::class.java).issues
|
objectMapper
|
||||||
|
.readValue(dataQualityReport, DataQualityReport::class.java)
|
||||||
|
.issues
|
||||||
|
.sortedBy { it.severity }
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
val otherIssue =
|
val otherIssue =
|
||||||
Issue(Severity.ERROR, "Not parsable data quality report '$dataQualityReport'")
|
Issue(Severity.ERROR, "Not parsable data quality report '$dataQualityReport'")
|
||||||
@ -54,7 +57,9 @@ class ReportService(
|
|||||||
data class Issue(val severity: Severity, val message: String)
|
data class Issue(val severity: Severity, val message: String)
|
||||||
|
|
||||||
enum class Severity(@JsonValue val value: String) {
|
enum class Severity(@JsonValue val value: String) {
|
||||||
|
FATAL("fatal"),
|
||||||
ERROR("error"),
|
ERROR("error"),
|
||||||
WARNING("warning"),
|
WARNING("warning"),
|
||||||
|
INFO("info")
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -20,10 +20,13 @@
|
|||||||
package dev.dnpm.etl.processor.monitoring
|
package dev.dnpm.etl.processor.monitoring
|
||||||
|
|
||||||
import org.springframework.data.annotation.Id
|
import org.springframework.data.annotation.Id
|
||||||
|
import org.springframework.data.domain.Page
|
||||||
|
import org.springframework.data.domain.Pageable
|
||||||
import org.springframework.data.jdbc.repository.query.Query
|
import org.springframework.data.jdbc.repository.query.Query
|
||||||
import org.springframework.data.relational.core.mapping.Embedded
|
import org.springframework.data.relational.core.mapping.Embedded
|
||||||
import org.springframework.data.relational.core.mapping.Table
|
import org.springframework.data.relational.core.mapping.Table
|
||||||
import org.springframework.data.repository.CrudRepository
|
import org.springframework.data.repository.CrudRepository
|
||||||
|
import org.springframework.data.repository.PagingAndSortingRepository
|
||||||
import java.time.Instant
|
import java.time.Instant
|
||||||
import java.util.*
|
import java.util.*
|
||||||
|
|
||||||
@ -52,12 +55,14 @@ data class CountedState(
|
|||||||
val status: RequestStatus,
|
val status: RequestStatus,
|
||||||
)
|
)
|
||||||
|
|
||||||
interface RequestRepository : CrudRepository<Request, Long> {
|
interface RequestRepository : CrudRepository<Request, Long>, PagingAndSortingRepository<Request, Long> {
|
||||||
|
|
||||||
fun findAllByPatientIdOrderByProcessedAtDesc(patientId: String): List<Request>
|
fun findAllByPatientIdOrderByProcessedAtDesc(patientId: String): List<Request>
|
||||||
|
|
||||||
fun findByUuidEquals(uuid: String): Optional<Request>
|
fun findByUuidEquals(uuid: String): Optional<Request>
|
||||||
|
|
||||||
|
fun findRequestByPatientId(patientId: String, pageable: Pageable): Page<Request>
|
||||||
|
|
||||||
@Query("SELECT count(*) AS count, status FROM request WHERE type = 'MTB_FILE' GROUP BY status ORDER BY status, count DESC;")
|
@Query("SELECT count(*) AS count, status FROM request WHERE type = 'MTB_FILE' GROUP BY status ORDER BY status, count DESC;")
|
||||||
fun countStates(): List<CountedState>
|
fun countStates(): List<CountedState>
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -22,14 +22,16 @@ package dev.dnpm.etl.processor.output
|
|||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
import de.ukw.ccc.bwhc.dto.Consent
|
import de.ukw.ccc.bwhc.dto.Consent
|
||||||
import de.ukw.ccc.bwhc.dto.MtbFile
|
import de.ukw.ccc.bwhc.dto.MtbFile
|
||||||
import dev.dnpm.etl.processor.config.KafkaTargetProperties
|
import dev.dnpm.etl.processor.config.KafkaProperties
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
||||||
import org.slf4j.LoggerFactory
|
import org.slf4j.LoggerFactory
|
||||||
import org.springframework.kafka.core.KafkaTemplate
|
import org.springframework.kafka.core.KafkaTemplate
|
||||||
|
import org.springframework.retry.support.RetryTemplate
|
||||||
|
|
||||||
class KafkaMtbFileSender(
|
class KafkaMtbFileSender(
|
||||||
private val kafkaTemplate: KafkaTemplate<String, String>,
|
private val kafkaTemplate: KafkaTemplate<String, String>,
|
||||||
private val kafkaTargetProperties: KafkaTargetProperties,
|
private val kafkaProperties: KafkaProperties,
|
||||||
|
private val retryTemplate: RetryTemplate,
|
||||||
private val objectMapper: ObjectMapper
|
private val objectMapper: ObjectMapper
|
||||||
) : MtbFileSender {
|
) : MtbFileSender {
|
||||||
|
|
||||||
@ -37,16 +39,18 @@ class KafkaMtbFileSender(
|
|||||||
|
|
||||||
override fun send(request: MtbFileSender.MtbFileRequest): MtbFileSender.Response {
|
override fun send(request: MtbFileSender.MtbFileRequest): MtbFileSender.Response {
|
||||||
return try {
|
return try {
|
||||||
val result = kafkaTemplate.send(
|
return retryTemplate.execute<MtbFileSender.Response, Exception> {
|
||||||
kafkaTargetProperties.topic,
|
val result = kafkaTemplate.send(
|
||||||
key(request),
|
kafkaProperties.topic,
|
||||||
objectMapper.writeValueAsString(request.mtbFile)
|
key(request),
|
||||||
)
|
objectMapper.writeValueAsString(Data(request.requestId, request.mtbFile))
|
||||||
if (result.get() != null) {
|
)
|
||||||
logger.debug("Sent file via KafkaMtbFileSender")
|
if (result.get() != null) {
|
||||||
MtbFileSender.Response(RequestStatus.UNKNOWN)
|
logger.debug("Sent file via KafkaMtbFileSender")
|
||||||
} else {
|
MtbFileSender.Response(RequestStatus.UNKNOWN)
|
||||||
MtbFileSender.Response(RequestStatus.ERROR)
|
} else {
|
||||||
|
MtbFileSender.Response(RequestStatus.ERROR)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
logger.error("An error occurred sending to kafka", e)
|
logger.error("An error occurred sending to kafka", e)
|
||||||
@ -65,17 +69,19 @@ class KafkaMtbFileSender(
|
|||||||
.build()
|
.build()
|
||||||
|
|
||||||
return try {
|
return try {
|
||||||
val result = kafkaTemplate.send(
|
return retryTemplate.execute<MtbFileSender.Response, Exception> {
|
||||||
kafkaTargetProperties.topic,
|
val result = kafkaTemplate.send(
|
||||||
key(request),
|
kafkaProperties.topic,
|
||||||
objectMapper.writeValueAsString(dummyMtbFile)
|
key(request),
|
||||||
)
|
objectMapper.writeValueAsString(Data(request.requestId, dummyMtbFile))
|
||||||
|
)
|
||||||
|
|
||||||
if (result.get() != null) {
|
if (result.get() != null) {
|
||||||
logger.debug("Sent deletion request via KafkaMtbFileSender")
|
logger.debug("Sent deletion request via KafkaMtbFileSender")
|
||||||
MtbFileSender.Response(RequestStatus.UNKNOWN)
|
MtbFileSender.Response(RequestStatus.UNKNOWN)
|
||||||
} else {
|
} else {
|
||||||
MtbFileSender.Response(RequestStatus.ERROR)
|
MtbFileSender.Response(RequestStatus.ERROR)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
logger.error("An error occurred sending to kafka", e)
|
logger.error("An error occurred sending to kafka", e)
|
||||||
@ -83,14 +89,17 @@ class KafkaMtbFileSender(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun endpoint(): String {
|
||||||
|
return "${this.kafkaProperties.servers} (${this.kafkaProperties.topic}/${this.kafkaProperties.responseTopic})"
|
||||||
|
}
|
||||||
|
|
||||||
private fun key(request: MtbFileSender.MtbFileRequest): String {
|
private fun key(request: MtbFileSender.MtbFileRequest): String {
|
||||||
return "{\"pid\": \"${request.mtbFile.patient.id}\", " +
|
return "{\"pid\": \"${request.mtbFile.patient.id}\"}"
|
||||||
"\"eid\": \"${request.mtbFile.episode.id}\", " +
|
|
||||||
"\"requestId\": \"${request.requestId}\"}"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun key(request: MtbFileSender.DeleteRequest): String {
|
private fun key(request: MtbFileSender.DeleteRequest): String {
|
||||||
return "{\"pid\": \"${request.patientId}\", " +
|
return "{\"pid\": \"${request.patientId}\"}"
|
||||||
"\"requestId\": \"${request.requestId}\"}"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
data class Data(val requestId: String, val content: MtbFile)
|
||||||
}
|
}
|
@ -28,6 +28,8 @@ interface MtbFileSender {
|
|||||||
|
|
||||||
fun send(request: DeleteRequest): Response
|
fun send(request: DeleteRequest): Response
|
||||||
|
|
||||||
|
fun endpoint(): String
|
||||||
|
|
||||||
data class Response(val status: RequestStatus, val body: String = "")
|
data class Response(val status: RequestStatus, val body: String = "")
|
||||||
|
|
||||||
data class MtbFileRequest(val requestId: String, val mtbFile: MtbFile)
|
data class MtbFileRequest(val requestId: String, val mtbFile: MtbFile)
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -25,32 +25,39 @@ import org.slf4j.LoggerFactory
|
|||||||
import org.springframework.http.HttpEntity
|
import org.springframework.http.HttpEntity
|
||||||
import org.springframework.http.HttpHeaders
|
import org.springframework.http.HttpHeaders
|
||||||
import org.springframework.http.MediaType
|
import org.springframework.http.MediaType
|
||||||
|
import org.springframework.retry.support.RetryTemplate
|
||||||
import org.springframework.web.client.RestClientException
|
import org.springframework.web.client.RestClientException
|
||||||
import org.springframework.web.client.RestTemplate
|
import org.springframework.web.client.RestTemplate
|
||||||
|
|
||||||
class RestMtbFileSender(
|
class RestMtbFileSender(
|
||||||
private val restTemplate: RestTemplate,
|
private val restTemplate: RestTemplate,
|
||||||
private val restTargetProperties: RestTargetProperties
|
private val restTargetProperties: RestTargetProperties,
|
||||||
|
private val retryTemplate: RetryTemplate
|
||||||
) : MtbFileSender {
|
) : MtbFileSender {
|
||||||
|
|
||||||
private val logger = LoggerFactory.getLogger(RestMtbFileSender::class.java)
|
private val logger = LoggerFactory.getLogger(RestMtbFileSender::class.java)
|
||||||
|
|
||||||
override fun send(request: MtbFileSender.MtbFileRequest): MtbFileSender.Response {
|
override fun send(request: MtbFileSender.MtbFileRequest): MtbFileSender.Response {
|
||||||
try {
|
try {
|
||||||
val headers = HttpHeaders()
|
return retryTemplate.execute<MtbFileSender.Response, Exception> {
|
||||||
headers.contentType = MediaType.APPLICATION_JSON
|
val headers = HttpHeaders()
|
||||||
val entityReq = HttpEntity(request.mtbFile, headers)
|
headers.contentType = MediaType.APPLICATION_JSON
|
||||||
val response = restTemplate.postForEntity(
|
val entityReq = HttpEntity(request.mtbFile, headers)
|
||||||
"${restTargetProperties.uri}/MTBFile",
|
val response = restTemplate.postForEntity(
|
||||||
entityReq,
|
"${restTargetProperties.uri}/MTBFile",
|
||||||
String::class.java
|
entityReq,
|
||||||
)
|
String::class.java
|
||||||
if (!response.statusCode.is2xxSuccessful) {
|
)
|
||||||
logger.warn("Error sending to remote system: {}", response.body)
|
if (!response.statusCode.is2xxSuccessful) {
|
||||||
return MtbFileSender.Response(response.statusCode.asRequestStatus(), "Status-Code: ${response.statusCode.value()}")
|
logger.warn("Error sending to remote system: {}", response.body)
|
||||||
|
return@execute MtbFileSender.Response(
|
||||||
|
response.statusCode.asRequestStatus(),
|
||||||
|
"Status-Code: ${response.statusCode.value()}"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
logger.debug("Sent file via RestMtbFileSender")
|
||||||
|
return@execute MtbFileSender.Response(response.statusCode.asRequestStatus(), response.body.orEmpty())
|
||||||
}
|
}
|
||||||
logger.debug("Sent file via RestMtbFileSender")
|
|
||||||
return MtbFileSender.Response(response.statusCode.asRequestStatus(), response.body.orEmpty())
|
|
||||||
} catch (e: IllegalArgumentException) {
|
} catch (e: IllegalArgumentException) {
|
||||||
logger.error("Not a valid URI to export to: '{}'", restTargetProperties.uri!!)
|
logger.error("Not a valid URI to export to: '{}'", restTargetProperties.uri!!)
|
||||||
} catch (e: RestClientException) {
|
} catch (e: RestClientException) {
|
||||||
@ -62,16 +69,18 @@ class RestMtbFileSender(
|
|||||||
|
|
||||||
override fun send(request: MtbFileSender.DeleteRequest): MtbFileSender.Response {
|
override fun send(request: MtbFileSender.DeleteRequest): MtbFileSender.Response {
|
||||||
try {
|
try {
|
||||||
val headers = HttpHeaders()
|
return retryTemplate.execute<MtbFileSender.Response, Exception> {
|
||||||
headers.contentType = MediaType.APPLICATION_JSON
|
val headers = HttpHeaders()
|
||||||
val entityReq = HttpEntity(null, headers)
|
headers.contentType = MediaType.APPLICATION_JSON
|
||||||
restTemplate.delete(
|
val entityReq = HttpEntity(null, headers)
|
||||||
"${restTargetProperties.uri}/Patient/${request.patientId}",
|
restTemplate.delete(
|
||||||
entityReq,
|
"${restTargetProperties.uri}/Patient/${request.patientId}",
|
||||||
String::class.java
|
entityReq,
|
||||||
)
|
String::class.java
|
||||||
logger.debug("Sent file via RestMtbFileSender")
|
)
|
||||||
return MtbFileSender.Response(RequestStatus.SUCCESS)
|
logger.debug("Sent file via RestMtbFileSender")
|
||||||
|
return@execute MtbFileSender.Response(RequestStatus.SUCCESS)
|
||||||
|
}
|
||||||
} catch (e: IllegalArgumentException) {
|
} catch (e: IllegalArgumentException) {
|
||||||
logger.error("Not a valid URI to export to: '{}'", restTargetProperties.uri!!)
|
logger.error("Not a valid URI to export to: '{}'", restTargetProperties.uri!!)
|
||||||
} catch (e: RestClientException) {
|
} catch (e: RestClientException) {
|
||||||
@ -81,4 +90,8 @@ class RestMtbFileSender(
|
|||||||
return MtbFileSender.Response(RequestStatus.ERROR, "Sonstiger Fehler bei der Übertragung")
|
return MtbFileSender.Response(RequestStatus.ERROR, "Sonstiger Fehler bei der Übertragung")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
override fun endpoint(): String {
|
||||||
|
return this.restTargetProperties.uri.orEmpty()
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -33,4 +33,8 @@ class PseudonymizeService(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun prefix(): String {
|
||||||
|
return configProperties.prefix
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -20,31 +20,206 @@
|
|||||||
package dev.dnpm.etl.processor.pseudonym
|
package dev.dnpm.etl.processor.pseudonym
|
||||||
|
|
||||||
import de.ukw.ccc.bwhc.dto.MtbFile
|
import de.ukw.ccc.bwhc.dto.MtbFile
|
||||||
|
import org.apache.commons.codec.digest.DigestUtils
|
||||||
|
|
||||||
|
/** Replaces patient ID with generated patient pseudonym
|
||||||
|
*
|
||||||
|
* @param pseudonymizeService The pseudonymizeService to be used
|
||||||
|
*
|
||||||
|
* @return The MTB file containing patient pseudonymes
|
||||||
|
*/
|
||||||
infix fun MtbFile.pseudonymizeWith(pseudonymizeService: PseudonymizeService) {
|
infix fun MtbFile.pseudonymizeWith(pseudonymizeService: PseudonymizeService) {
|
||||||
val patientPseudonym = pseudonymizeService.patientPseudonym(this.patient.id)
|
val patientPseudonym = pseudonymizeService.patientPseudonym(this.patient.id)
|
||||||
|
|
||||||
this.episode.patient = patientPseudonym
|
this.episode?.patient = patientPseudonym
|
||||||
this.carePlans.forEach { it.patient = patientPseudonym }
|
this.carePlans?.forEach { it.patient = patientPseudonym }
|
||||||
this.patient.id = patientPseudonym
|
this.patient.id = patientPseudonym
|
||||||
this.claims.forEach { it.patient = patientPseudonym }
|
this.claims?.forEach { it.patient = patientPseudonym }
|
||||||
this.consent.patient = patientPseudonym
|
this.consent?.patient = patientPseudonym
|
||||||
this.claimResponses.forEach { it.patient = patientPseudonym }
|
this.claimResponses?.forEach { it.patient = patientPseudonym }
|
||||||
this.diagnoses.forEach { it.patient = patientPseudonym }
|
this.diagnoses?.forEach { it.patient = patientPseudonym }
|
||||||
this.ecogStatus.forEach { it.patient = patientPseudonym }
|
this.ecogStatus?.forEach { it.patient = patientPseudonym }
|
||||||
this.familyMemberDiagnoses.forEach { it.patient = patientPseudonym }
|
this.familyMemberDiagnoses?.forEach { it.patient = patientPseudonym }
|
||||||
this.geneticCounsellingRequests.forEach { it.patient = patientPseudonym }
|
this.geneticCounsellingRequests?.forEach { it.patient = patientPseudonym }
|
||||||
this.histologyReevaluationRequests.forEach { it.patient = patientPseudonym }
|
this.histologyReevaluationRequests?.forEach { it.patient = patientPseudonym }
|
||||||
this.histologyReports.forEach { it.patient = patientPseudonym }
|
this.histologyReports?.forEach {
|
||||||
this.lastGuidelineTherapies.forEach { it.patient = patientPseudonym }
|
it.patient = patientPseudonym
|
||||||
this.molecularPathologyFindings.forEach { it.patient = patientPseudonym }
|
it.tumorMorphology?.patient = patientPseudonym
|
||||||
this.molecularTherapies.forEach { it.history.forEach { it.patient = patientPseudonym } }
|
}
|
||||||
this.ngsReports.forEach { it.patient = patientPseudonym }
|
this.lastGuidelineTherapies?.forEach { it.patient = patientPseudonym }
|
||||||
this.previousGuidelineTherapies.forEach { it.patient = patientPseudonym }
|
this.molecularPathologyFindings?.forEach { it.patient = patientPseudonym }
|
||||||
this.rebiopsyRequests.forEach { it.patient = patientPseudonym }
|
this.molecularTherapies?.forEach { molecularTherapy -> molecularTherapy.history.forEach { it.patient = patientPseudonym } }
|
||||||
this.recommendations.forEach { it.patient = patientPseudonym }
|
this.ngsReports?.forEach { it.patient = patientPseudonym }
|
||||||
this.recommendations.forEach { it.patient = patientPseudonym }
|
this.previousGuidelineTherapies?.forEach { it.patient = patientPseudonym }
|
||||||
this.responses.forEach { it.patient = patientPseudonym }
|
this.rebiopsyRequests?.forEach { it.patient = patientPseudonym }
|
||||||
this.specimens.forEach { it.patient = patientPseudonym }
|
this.recommendations?.forEach { it.patient = patientPseudonym }
|
||||||
this.specimens.forEach { it.patient = patientPseudonym }
|
this.responses?.forEach { it.patient = patientPseudonym }
|
||||||
|
this.studyInclusionRequests?.forEach { it.patient = patientPseudonym }
|
||||||
|
this.specimens?.forEach { it.patient = patientPseudonym }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates new hash of content IDs with given prefix except for patient IDs
|
||||||
|
*
|
||||||
|
* @param pseudonymizeService The pseudonymizeService to be used
|
||||||
|
*
|
||||||
|
* @return The MTB file containing rehashed content IDs
|
||||||
|
*/
|
||||||
|
infix fun MtbFile.anonymizeContentWith(pseudonymizeService: PseudonymizeService) {
|
||||||
|
val prefix = pseudonymizeService.prefix()
|
||||||
|
|
||||||
|
fun anonymize(id: String): String {
|
||||||
|
val hash = DigestUtils.sha256Hex("$prefix-$id").substring(0, 41).lowercase()
|
||||||
|
return "$prefix$hash"
|
||||||
|
}
|
||||||
|
|
||||||
|
this.episode?.apply {
|
||||||
|
id = id?.let {
|
||||||
|
anonymize(it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.carePlans?.onEach { carePlan ->
|
||||||
|
carePlan?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
diagnosis = diagnosis?.let { anonymize(it) }
|
||||||
|
geneticCounsellingRequest = geneticCounsellingRequest?.let { anonymize(it) }
|
||||||
|
rebiopsyRequests = rebiopsyRequests.map { it?.let { anonymize(it) } }
|
||||||
|
recommendations = recommendations.map { it?.let { anonymize(it) } }
|
||||||
|
studyInclusionRequests = studyInclusionRequests.map { it?.let { anonymize(it) } }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.claims?.onEach { claim ->
|
||||||
|
claim?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
therapy = therapy?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.claimResponses?.onEach { claimResponse ->
|
||||||
|
claimResponse?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
claim = claim?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.consent?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
this.diagnoses?.onEach { diagnosis ->
|
||||||
|
diagnosis?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
histologyResults = histologyResults?.map { it?.let { anonymize(it) } }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.ecogStatus?.onEach { ecogStatus ->
|
||||||
|
ecogStatus?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.familyMemberDiagnoses?.onEach { familyMemberDiagnosis ->
|
||||||
|
familyMemberDiagnosis?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.geneticCounsellingRequests?.onEach { geneticCounsellingRequest ->
|
||||||
|
geneticCounsellingRequest?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.histologyReevaluationRequests?.onEach { histologyReevaluationRequest ->
|
||||||
|
histologyReevaluationRequest?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
specimen = specimen?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.histologyReports?.onEach { histologyReport ->
|
||||||
|
histologyReport?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
specimen = specimen?.let { anonymize(it) }
|
||||||
|
tumorMorphology?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
specimen = specimen?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
tumorCellContent?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
specimen = specimen?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.lastGuidelineTherapies?.onEach { lastGuidelineTherapy ->
|
||||||
|
lastGuidelineTherapy?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
diagnosis = diagnosis?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.molecularPathologyFindings?.onEach { molecularPathologyFinding ->
|
||||||
|
molecularPathologyFinding?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
specimen = specimen?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.molecularTherapies?.onEach { molecularTherapy ->
|
||||||
|
molecularTherapy?.apply {
|
||||||
|
history?.onEach { history ->
|
||||||
|
history?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
basedOn = basedOn?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.ngsReports?.onEach { ngsReport ->
|
||||||
|
ngsReport?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
specimen = specimen?.let { anonymize(it) }
|
||||||
|
tumorCellContent?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
specimen = specimen?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
simpleVariants?.onEach { simpleVariant ->
|
||||||
|
simpleVariant?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.previousGuidelineTherapies?.onEach { previousGuidelineTherapy ->
|
||||||
|
previousGuidelineTherapy?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
diagnosis = diagnosis?.let { anonymize(it) }
|
||||||
|
medication.forEach { medication ->
|
||||||
|
medication?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.rebiopsyRequests?.onEach { rebiopsyRequest ->
|
||||||
|
rebiopsyRequest?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
specimen = specimen?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.recommendations?.onEach { recommendation ->
|
||||||
|
recommendation?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
diagnosis = diagnosis?.let { anonymize(it) }
|
||||||
|
ngsReport = ngsReport?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.responses?.onEach { response ->
|
||||||
|
response?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
therapy = therapy?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.studyInclusionRequests?.onEach { studyInclusionRequest ->
|
||||||
|
studyInclusionRequest?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
reason = reason?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.specimens?.onEach { specimen ->
|
||||||
|
specimen?.apply {
|
||||||
|
id = id?.let { anonymize(it) }
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
45
src/main/kotlin/dev/dnpm/etl/processor/security/UserRole.kt
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (C) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.security
|
||||||
|
|
||||||
|
import org.springframework.data.annotation.Id
|
||||||
|
import org.springframework.data.relational.core.mapping.Table
|
||||||
|
import org.springframework.data.repository.CrudRepository
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
|
@Table("user_role")
|
||||||
|
data class UserRole(
|
||||||
|
@Id val id: Long? = null,
|
||||||
|
val username: String,
|
||||||
|
var role: Role = Role.GUEST
|
||||||
|
)
|
||||||
|
|
||||||
|
enum class Role(val value: String) {
|
||||||
|
GUEST("guest"),
|
||||||
|
USER("user"),
|
||||||
|
ADMIN("admin")
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UserRoleRepository : CrudRepository<UserRole, Long> {
|
||||||
|
|
||||||
|
fun findByUsername(username: String): Optional<UserRole>
|
||||||
|
|
||||||
|
}
|
@ -21,16 +21,17 @@ package dev.dnpm.etl.processor.services
|
|||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
import de.ukw.ccc.bwhc.dto.MtbFile
|
import de.ukw.ccc.bwhc.dto.MtbFile
|
||||||
|
import dev.dnpm.etl.processor.config.AppConfigProperties
|
||||||
import dev.dnpm.etl.processor.monitoring.Report
|
import dev.dnpm.etl.processor.monitoring.Report
|
||||||
import dev.dnpm.etl.processor.monitoring.Request
|
import dev.dnpm.etl.processor.monitoring.Request
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestType
|
import dev.dnpm.etl.processor.monitoring.RequestType
|
||||||
import dev.dnpm.etl.processor.output.MtbFileSender
|
import dev.dnpm.etl.processor.output.MtbFileSender
|
||||||
import dev.dnpm.etl.processor.pseudonym.PseudonymizeService
|
import dev.dnpm.etl.processor.pseudonym.PseudonymizeService
|
||||||
|
import dev.dnpm.etl.processor.pseudonym.anonymizeContentWith
|
||||||
import dev.dnpm.etl.processor.pseudonym.pseudonymizeWith
|
import dev.dnpm.etl.processor.pseudonym.pseudonymizeWith
|
||||||
import org.apache.commons.codec.binary.Base32
|
import org.apache.commons.codec.binary.Base32
|
||||||
import org.apache.commons.codec.digest.DigestUtils
|
import org.apache.commons.codec.digest.DigestUtils
|
||||||
import org.slf4j.LoggerFactory
|
|
||||||
import org.springframework.context.ApplicationEventPublisher
|
import org.springframework.context.ApplicationEventPublisher
|
||||||
import org.springframework.stereotype.Service
|
import org.springframework.stereotype.Service
|
||||||
import java.time.Instant
|
import java.time.Instant
|
||||||
@ -39,21 +40,25 @@ import java.util.*
|
|||||||
@Service
|
@Service
|
||||||
class RequestProcessor(
|
class RequestProcessor(
|
||||||
private val pseudonymizeService: PseudonymizeService,
|
private val pseudonymizeService: PseudonymizeService,
|
||||||
|
private val transformationService: TransformationService,
|
||||||
private val sender: MtbFileSender,
|
private val sender: MtbFileSender,
|
||||||
private val requestService: RequestService,
|
private val requestService: RequestService,
|
||||||
private val objectMapper: ObjectMapper,
|
private val objectMapper: ObjectMapper,
|
||||||
private val applicationEventPublisher: ApplicationEventPublisher
|
private val applicationEventPublisher: ApplicationEventPublisher,
|
||||||
|
private val appConfigProperties: AppConfigProperties
|
||||||
) {
|
) {
|
||||||
|
|
||||||
private val logger = LoggerFactory.getLogger(RequestProcessor::class.java)
|
|
||||||
|
|
||||||
fun processMtbFile(mtbFile: MtbFile) {
|
fun processMtbFile(mtbFile: MtbFile) {
|
||||||
val requestId = UUID.randomUUID().toString()
|
processMtbFile(mtbFile, UUID.randomUUID().toString())
|
||||||
|
}
|
||||||
|
|
||||||
|
fun processMtbFile(mtbFile: MtbFile, requestId: String) {
|
||||||
val pid = mtbFile.patient.id
|
val pid = mtbFile.patient.id
|
||||||
|
|
||||||
mtbFile pseudonymizeWith pseudonymizeService
|
mtbFile pseudonymizeWith pseudonymizeService
|
||||||
|
mtbFile anonymizeContentWith pseudonymizeService
|
||||||
|
|
||||||
val request = MtbFileSender.MtbFileRequest(requestId, mtbFile)
|
val request = MtbFileSender.MtbFileRequest(requestId, transformationService.transform(mtbFile))
|
||||||
|
|
||||||
requestService.save(
|
requestService.save(
|
||||||
Request(
|
Request(
|
||||||
@ -66,7 +71,7 @@ class RequestProcessor(
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
if (isDuplication(mtbFile)) {
|
if (appConfigProperties.duplicationDetection && isDuplication(mtbFile)) {
|
||||||
applicationEventPublisher.publishEvent(
|
applicationEventPublisher.publishEvent(
|
||||||
ResponseEvent(
|
ResponseEvent(
|
||||||
requestId,
|
requestId,
|
||||||
@ -103,8 +108,10 @@ class RequestProcessor(
|
|||||||
}
|
}
|
||||||
|
|
||||||
fun processDeletion(patientId: String) {
|
fun processDeletion(patientId: String) {
|
||||||
val requestId = UUID.randomUUID().toString()
|
processDeletion(patientId, UUID.randomUUID().toString())
|
||||||
|
}
|
||||||
|
|
||||||
|
fun processDeletion(patientId: String, requestId: String) {
|
||||||
try {
|
try {
|
||||||
val patientPseudonym = pseudonymizeService.patientPseudonym(patientId)
|
val patientPseudonym = pseudonymizeService.patientPseudonym(patientId)
|
||||||
|
|
||||||
|
@ -19,7 +19,6 @@
|
|||||||
|
|
||||||
package dev.dnpm.etl.processor.services
|
package dev.dnpm.etl.processor.services
|
||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
|
||||||
import dev.dnpm.etl.processor.monitoring.Report
|
import dev.dnpm.etl.processor.monitoring.Report
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
||||||
@ -33,8 +32,7 @@ import java.util.*
|
|||||||
@Service
|
@Service
|
||||||
class ResponseProcessor(
|
class ResponseProcessor(
|
||||||
private val requestRepository: RequestRepository,
|
private val requestRepository: RequestRepository,
|
||||||
private val statisticsUpdateProducer: Sinks.Many<Any>,
|
private val statisticsUpdateProducer: Sinks.Many<Any>
|
||||||
private val objectMapper: ObjectMapper
|
|
||||||
) {
|
) {
|
||||||
|
|
||||||
private val logger = LoggerFactory.getLogger(ResponseProcessor::class.java)
|
private val logger = LoggerFactory.getLogger(ResponseProcessor::class.java)
|
||||||
@ -73,7 +71,7 @@ class ResponseProcessor(
|
|||||||
}
|
}
|
||||||
|
|
||||||
else -> {
|
else -> {
|
||||||
logger.error("Cannot process response: Unknown response code!")
|
logger.error("Cannot process response: Unknown response!")
|
||||||
return@ifPresentOrElse
|
return@ifPresentOrElse
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,92 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.services
|
||||||
|
|
||||||
|
import jakarta.annotation.PostConstruct
|
||||||
|
import org.springframework.data.annotation.Id
|
||||||
|
import org.springframework.data.relational.core.mapping.Table
|
||||||
|
import org.springframework.data.repository.CrudRepository
|
||||||
|
import org.springframework.data.repository.findByIdOrNull
|
||||||
|
import org.springframework.security.core.userdetails.User
|
||||||
|
import org.springframework.security.crypto.password.PasswordEncoder
|
||||||
|
import org.springframework.security.provisioning.InMemoryUserDetailsManager
|
||||||
|
import java.time.Instant
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
|
class TokenService(
|
||||||
|
private val userDetailsManager: InMemoryUserDetailsManager,
|
||||||
|
private val passwordEncoder: PasswordEncoder,
|
||||||
|
private val tokenRepository: TokenRepository
|
||||||
|
) {
|
||||||
|
|
||||||
|
@PostConstruct
|
||||||
|
fun setup() {
|
||||||
|
tokenRepository.findAll().forEach {
|
||||||
|
userDetailsManager.createUser(
|
||||||
|
User.withUsername(it.username)
|
||||||
|
.password(it.password)
|
||||||
|
.roles("MTBFILE")
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun addToken(name: String): Result<String> {
|
||||||
|
val username = name.lowercase().replace("""[^a-z0-9]""".toRegex(), "")
|
||||||
|
if (userDetailsManager.userExists(username)) {
|
||||||
|
return Result.failure(RuntimeException("Cannot use token name"))
|
||||||
|
}
|
||||||
|
|
||||||
|
val password = Base64.getEncoder().encodeToString(UUID.randomUUID().toString().encodeToByteArray())
|
||||||
|
val encodedPassword = passwordEncoder.encode(password).toString()
|
||||||
|
|
||||||
|
userDetailsManager.createUser(
|
||||||
|
User.withUsername(username)
|
||||||
|
.password(encodedPassword)
|
||||||
|
.roles("MTBFILE")
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
|
||||||
|
tokenRepository.save(Token(name = name, username = username, password = encodedPassword))
|
||||||
|
|
||||||
|
return Result.success("$username:$password")
|
||||||
|
}
|
||||||
|
|
||||||
|
fun deleteToken(id: Long) {
|
||||||
|
val token = tokenRepository.findByIdOrNull(id) ?: return
|
||||||
|
userDetailsManager.deleteUser(token.username)
|
||||||
|
tokenRepository.delete(token)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun findAll(): List<Token> {
|
||||||
|
return tokenRepository.findAll().toList()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Table("token")
|
||||||
|
data class Token(
|
||||||
|
@Id val id: Long? = null,
|
||||||
|
val name: String,
|
||||||
|
val username: String,
|
||||||
|
val password: String,
|
||||||
|
val createdAt: Instant = Instant.now()
|
||||||
|
)
|
||||||
|
|
||||||
|
interface TokenRepository : CrudRepository<Token, Long>
|
@ -0,0 +1,85 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.services
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import com.jayway.jsonpath.JsonPath
|
||||||
|
import com.jayway.jsonpath.PathNotFoundException
|
||||||
|
import de.ukw.ccc.bwhc.dto.MtbFile
|
||||||
|
|
||||||
|
class TransformationService(private val objectMapper: ObjectMapper, private val transformations: List<Transformation>) {
|
||||||
|
fun transform(mtbFile: MtbFile): MtbFile {
|
||||||
|
var json = objectMapper.writeValueAsString(mtbFile)
|
||||||
|
|
||||||
|
transformations.forEach { transformation ->
|
||||||
|
val jsonPath = JsonPath.parse(json)
|
||||||
|
|
||||||
|
try {
|
||||||
|
val before = transformation.path.substringBeforeLast(".")
|
||||||
|
val last = transformation.path.substringAfterLast(".")
|
||||||
|
|
||||||
|
val existingValue = if (transformation.existingValue is Number) transformation.existingValue else transformation.existingValue.toString()
|
||||||
|
val newValue = if (transformation.newValue is Number) transformation.newValue else transformation.newValue.toString()
|
||||||
|
|
||||||
|
jsonPath.set("$.$before.[?]$last", newValue, {
|
||||||
|
it.item(HashMap::class.java)[last] == existingValue
|
||||||
|
})
|
||||||
|
} catch (e: PathNotFoundException) {
|
||||||
|
// Ignore
|
||||||
|
}
|
||||||
|
|
||||||
|
json = jsonPath.jsonString()
|
||||||
|
}
|
||||||
|
|
||||||
|
return objectMapper.readValue(json, MtbFile::class.java)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun getTransformations(): List<Transformation> {
|
||||||
|
return this.transformations
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class Transformation private constructor(val path: String) {
|
||||||
|
|
||||||
|
lateinit var existingValue: Any
|
||||||
|
private set
|
||||||
|
lateinit var newValue: Any
|
||||||
|
private set
|
||||||
|
|
||||||
|
infix fun from(value: Any): Transformation {
|
||||||
|
this.existingValue = value
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
infix fun to(value: Any): Transformation {
|
||||||
|
this.newValue = value
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
|
||||||
|
fun of(path: String): Transformation {
|
||||||
|
return Transformation(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,61 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.services
|
||||||
|
|
||||||
|
import dev.dnpm.etl.processor.security.Role
|
||||||
|
import dev.dnpm.etl.processor.security.UserRole
|
||||||
|
import dev.dnpm.etl.processor.security.UserRoleRepository
|
||||||
|
import org.springframework.data.repository.findByIdOrNull
|
||||||
|
import org.springframework.security.core.session.SessionRegistry
|
||||||
|
import org.springframework.security.oauth2.core.oidc.user.OidcUser
|
||||||
|
|
||||||
|
class UserRoleService(
|
||||||
|
private val userRoleRepository: UserRoleRepository,
|
||||||
|
private val sessionRegistry: SessionRegistry
|
||||||
|
) {
|
||||||
|
fun updateUserRole(id: Long, role: Role) {
|
||||||
|
val userRole = userRoleRepository.findByIdOrNull(id) ?: return
|
||||||
|
userRole.role = role
|
||||||
|
userRoleRepository.save(userRole)
|
||||||
|
expireSessionFor(userRole.username)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun deleteUserRole(id: Long) {
|
||||||
|
val userRole = userRoleRepository.findByIdOrNull(id) ?: return
|
||||||
|
userRoleRepository.delete(userRole)
|
||||||
|
expireSessionFor(userRole.username)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun findAll(): List<UserRole> {
|
||||||
|
return userRoleRepository.findAll().toList()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun expireSessionFor(username: String) {
|
||||||
|
sessionRegistry.allPrincipals
|
||||||
|
.filterIsInstance<OidcUser>()
|
||||||
|
.filter { it.preferredUsername == username }
|
||||||
|
.flatMap {
|
||||||
|
sessionRegistry.getAllSessions(it, true)
|
||||||
|
}
|
||||||
|
.onEach {
|
||||||
|
it.expireNow()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -41,50 +41,40 @@ class KafkaResponseProcessor(
|
|||||||
|
|
||||||
override fun onMessage(data: ConsumerRecord<String, String>) {
|
override fun onMessage(data: ConsumerRecord<String, String>) {
|
||||||
try {
|
try {
|
||||||
Optional.of(objectMapper.readValue(data.key(), ResponseKey::class.java))
|
Optional.of(objectMapper.readValue(data.value(), ResponseBody::class.java))
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
|
logger.error("Cannot process Kafka response", e)
|
||||||
Optional.empty()
|
Optional.empty()
|
||||||
}.ifPresentOrElse({ responseKey ->
|
}.ifPresentOrElse({ responseBody ->
|
||||||
val event = try {
|
val event = ResponseEvent(
|
||||||
val responseBody = objectMapper.readValue(data.value(), ResponseBody::class.java)
|
responseBody.requestId,
|
||||||
ResponseEvent(
|
Instant.ofEpochMilli(data.timestamp()),
|
||||||
responseKey.requestId,
|
responseBody.statusCode.asRequestStatus(),
|
||||||
Instant.ofEpochMilli(data.timestamp()),
|
when (responseBody.statusCode.asRequestStatus()) {
|
||||||
responseBody.statusCode.asRequestStatus(),
|
RequestStatus.SUCCESS -> {
|
||||||
when (responseBody.statusCode.asRequestStatus()) {
|
Optional.empty()
|
||||||
RequestStatus.SUCCESS -> {
|
|
||||||
Optional.empty()
|
|
||||||
}
|
|
||||||
|
|
||||||
RequestStatus.WARNING, RequestStatus.ERROR -> {
|
|
||||||
Optional.of(objectMapper.writeValueAsString(responseBody.statusBody))
|
|
||||||
}
|
|
||||||
|
|
||||||
else -> {
|
|
||||||
logger.error("Kafka response: Unknown response code!")
|
|
||||||
Optional.empty()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
)
|
|
||||||
} catch (e: Exception) {
|
RequestStatus.WARNING, RequestStatus.ERROR -> {
|
||||||
logger.error("Cannot process Kafka response", e)
|
Optional.of(objectMapper.writeValueAsString(responseBody.statusBody))
|
||||||
ResponseEvent(
|
}
|
||||||
responseKey.requestId,
|
|
||||||
Instant.ofEpochMilli(data.timestamp()),
|
else -> {
|
||||||
RequestStatus.ERROR,
|
logger.error("Kafka response: Unknown response code '{}'!", responseBody.statusCode)
|
||||||
Optional.of("Cannot process Kafka response")
|
Optional.empty()
|
||||||
)
|
}
|
||||||
}
|
}
|
||||||
|
)
|
||||||
eventPublisher.publishEvent(event)
|
eventPublisher.publishEvent(event)
|
||||||
}, {
|
}, {
|
||||||
logger.error("No response key in Kafka response")
|
logger.error("No requestId in Kafka response")
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
data class ResponseKey(val requestId: String)
|
|
||||||
|
|
||||||
data class ResponseBody(
|
data class ResponseBody(
|
||||||
@JsonProperty("status_code") @JsonAlias("status code") val statusCode: Int,
|
@JsonProperty("request_id") @JsonAlias("requestId") val requestId: String,
|
||||||
@JsonProperty("status_body") val statusBody: Map<String, Any>
|
@JsonProperty("status_code") @JsonAlias("statusCode") val statusCode: Int,
|
||||||
|
@JsonProperty("status_body") @JsonAlias("statusBody") val statusBody: Map<String, Any>
|
||||||
)
|
)
|
||||||
|
|
||||||
}
|
}
|
199
src/main/kotlin/dev/dnpm/etl/processor/web/ConfigController.kt
Normal file
@ -0,0 +1,199 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.web
|
||||||
|
|
||||||
|
import dev.dnpm.etl.processor.monitoring.ConnectionCheckResult
|
||||||
|
import dev.dnpm.etl.processor.monitoring.ConnectionCheckService
|
||||||
|
import dev.dnpm.etl.processor.monitoring.GPasConnectionCheckService
|
||||||
|
import dev.dnpm.etl.processor.monitoring.OutputConnectionCheckService
|
||||||
|
import dev.dnpm.etl.processor.output.MtbFileSender
|
||||||
|
import dev.dnpm.etl.processor.pseudonym.Generator
|
||||||
|
import dev.dnpm.etl.processor.security.Role
|
||||||
|
import dev.dnpm.etl.processor.security.UserRole
|
||||||
|
import dev.dnpm.etl.processor.services.Token
|
||||||
|
import dev.dnpm.etl.processor.services.TokenService
|
||||||
|
import dev.dnpm.etl.processor.services.TransformationService
|
||||||
|
import dev.dnpm.etl.processor.services.UserRoleService
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier
|
||||||
|
import org.springframework.http.MediaType
|
||||||
|
import org.springframework.http.codec.ServerSentEvent
|
||||||
|
import org.springframework.stereotype.Controller
|
||||||
|
import org.springframework.ui.Model
|
||||||
|
import org.springframework.web.bind.annotation.*
|
||||||
|
import reactor.core.publisher.Flux
|
||||||
|
import reactor.core.publisher.Sinks
|
||||||
|
|
||||||
|
@Controller
|
||||||
|
@RequestMapping(path = ["configs"])
|
||||||
|
class ConfigController(
|
||||||
|
@Qualifier("connectionCheckUpdateProducer")
|
||||||
|
private val connectionCheckUpdateProducer: Sinks.Many<ConnectionCheckResult>,
|
||||||
|
private val transformationService: TransformationService,
|
||||||
|
private val pseudonymGenerator: Generator,
|
||||||
|
private val mtbFileSender: MtbFileSender,
|
||||||
|
private val connectionCheckServices: List<ConnectionCheckService>,
|
||||||
|
private val tokenService: TokenService?,
|
||||||
|
private val userRoleService: UserRoleService?
|
||||||
|
) {
|
||||||
|
|
||||||
|
@GetMapping
|
||||||
|
fun index(model: Model): String {
|
||||||
|
val outputConnectionAvailable =
|
||||||
|
connectionCheckServices.filterIsInstance<OutputConnectionCheckService>().first().connectionAvailable()
|
||||||
|
|
||||||
|
val gPasConnectionAvailable =
|
||||||
|
connectionCheckServices.filterIsInstance<GPasConnectionCheckService>().firstOrNull()?.connectionAvailable()
|
||||||
|
|
||||||
|
model.addAttribute("pseudonymGenerator", pseudonymGenerator.javaClass.simpleName)
|
||||||
|
model.addAttribute("mtbFileSender", mtbFileSender.javaClass.simpleName)
|
||||||
|
model.addAttribute("mtbFileEndpoint", mtbFileSender.endpoint())
|
||||||
|
model.addAttribute("outputConnectionAvailable", outputConnectionAvailable)
|
||||||
|
model.addAttribute("gPasConnectionAvailable", gPasConnectionAvailable)
|
||||||
|
model.addAttribute("tokensEnabled", tokenService != null)
|
||||||
|
if (tokenService != null) {
|
||||||
|
model.addAttribute("tokens", tokenService.findAll())
|
||||||
|
} else {
|
||||||
|
model.addAttribute("tokens", emptyList<Token>())
|
||||||
|
}
|
||||||
|
model.addAttribute("transformations", transformationService.getTransformations())
|
||||||
|
if (userRoleService != null) {
|
||||||
|
model.addAttribute("userRolesEnabled", true)
|
||||||
|
model.addAttribute("userRoles", userRoleService.findAll())
|
||||||
|
} else {
|
||||||
|
model.addAttribute("userRolesEnabled", false)
|
||||||
|
model.addAttribute("userRoles", emptyList<UserRole>())
|
||||||
|
}
|
||||||
|
return "configs"
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping(params = ["outputConnectionAvailable"])
|
||||||
|
fun outputConnectionAvailable(model: Model): String {
|
||||||
|
val outputConnectionAvailable =
|
||||||
|
connectionCheckServices.filterIsInstance<OutputConnectionCheckService>().first().connectionAvailable()
|
||||||
|
|
||||||
|
model.addAttribute("mtbFileSender", mtbFileSender.javaClass.simpleName)
|
||||||
|
model.addAttribute("mtbFileEndpoint", mtbFileSender.endpoint())
|
||||||
|
model.addAttribute("outputConnectionAvailable", outputConnectionAvailable)
|
||||||
|
if (tokenService != null) {
|
||||||
|
model.addAttribute("tokensEnabled", true)
|
||||||
|
model.addAttribute("tokens", tokenService.findAll())
|
||||||
|
} else {
|
||||||
|
model.addAttribute("tokens", listOf<Token>())
|
||||||
|
}
|
||||||
|
|
||||||
|
return "configs/outputConnectionAvailable"
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping(params = ["gPasConnectionAvailable"])
|
||||||
|
fun gPasConnectionAvailable(model: Model): String {
|
||||||
|
val gPasConnectionAvailable =
|
||||||
|
connectionCheckServices.filterIsInstance<GPasConnectionCheckService>().firstOrNull()?.connectionAvailable()
|
||||||
|
|
||||||
|
model.addAttribute("mtbFileSender", mtbFileSender.javaClass.simpleName)
|
||||||
|
model.addAttribute("mtbFileEndpoint", mtbFileSender.endpoint())
|
||||||
|
model.addAttribute("gPasConnectionAvailable", gPasConnectionAvailable)
|
||||||
|
if (tokenService != null) {
|
||||||
|
model.addAttribute("tokensEnabled", true)
|
||||||
|
model.addAttribute("tokens", tokenService.findAll())
|
||||||
|
} else {
|
||||||
|
model.addAttribute("tokens", listOf<Token>())
|
||||||
|
}
|
||||||
|
|
||||||
|
return "configs/gPasConnectionAvailable"
|
||||||
|
}
|
||||||
|
|
||||||
|
@PostMapping(path = ["tokens"])
|
||||||
|
fun addToken(@ModelAttribute("name") name: String, model: Model): String {
|
||||||
|
if (tokenService == null) {
|
||||||
|
model.addAttribute("tokensEnabled", false)
|
||||||
|
model.addAttribute("success", false)
|
||||||
|
} else {
|
||||||
|
model.addAttribute("tokensEnabled", true)
|
||||||
|
val result = tokenService.addToken(name)
|
||||||
|
if (result.isSuccess) {
|
||||||
|
model.addAttribute("newTokenValue", result.getOrDefault(""))
|
||||||
|
model.addAttribute("success", true)
|
||||||
|
} else {
|
||||||
|
model.addAttribute("success", false)
|
||||||
|
}
|
||||||
|
model.addAttribute("tokens", tokenService.findAll())
|
||||||
|
}
|
||||||
|
|
||||||
|
return "configs/tokens"
|
||||||
|
}
|
||||||
|
|
||||||
|
@DeleteMapping(path = ["tokens/{id}"])
|
||||||
|
fun deleteToken(@PathVariable id: Long, model: Model): String {
|
||||||
|
if (tokenService != null) {
|
||||||
|
tokenService.deleteToken(id)
|
||||||
|
|
||||||
|
model.addAttribute("tokensEnabled", true)
|
||||||
|
model.addAttribute("tokens", tokenService.findAll())
|
||||||
|
} else {
|
||||||
|
model.addAttribute("tokensEnabled", false)
|
||||||
|
model.addAttribute("tokens", listOf<Token>())
|
||||||
|
}
|
||||||
|
return "configs/tokens"
|
||||||
|
}
|
||||||
|
|
||||||
|
@DeleteMapping(path = ["userroles/{id}"])
|
||||||
|
fun deleteUserRole(@PathVariable id: Long, model: Model): String {
|
||||||
|
if (userRoleService != null) {
|
||||||
|
userRoleService.deleteUserRole(id)
|
||||||
|
|
||||||
|
model.addAttribute("userRolesEnabled", true)
|
||||||
|
model.addAttribute("userRoles", userRoleService.findAll())
|
||||||
|
} else {
|
||||||
|
model.addAttribute("userRolesEnabled", false)
|
||||||
|
model.addAttribute("userRoles", emptyList<UserRole>())
|
||||||
|
}
|
||||||
|
return "configs/userroles"
|
||||||
|
}
|
||||||
|
|
||||||
|
@PutMapping(path = ["userroles/{id}"])
|
||||||
|
fun updateUserRole(@PathVariable id: Long, @ModelAttribute("role") role: Role, model: Model): String {
|
||||||
|
if (userRoleService != null) {
|
||||||
|
userRoleService.updateUserRole(id, role)
|
||||||
|
|
||||||
|
model.addAttribute("userRolesEnabled", true)
|
||||||
|
model.addAttribute("userRoles", userRoleService.findAll())
|
||||||
|
} else {
|
||||||
|
model.addAttribute("userRolesEnabled", false)
|
||||||
|
model.addAttribute("userRoles", emptyList<UserRole>())
|
||||||
|
}
|
||||||
|
return "configs/userroles"
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping(path = ["events"], produces = [MediaType.TEXT_EVENT_STREAM_VALUE])
|
||||||
|
fun events(): Flux<ServerSentEvent<Any>> {
|
||||||
|
return connectionCheckUpdateProducer.asFlux().map {
|
||||||
|
val event = when (it) {
|
||||||
|
is ConnectionCheckResult.KafkaConnectionCheckResult -> "output-connection-check"
|
||||||
|
is ConnectionCheckResult.RestConnectionCheckResult -> "output-connection-check"
|
||||||
|
is ConnectionCheckResult.GPasConnectionCheckResult -> "gpas-connection-check"
|
||||||
|
}
|
||||||
|
|
||||||
|
ServerSentEvent.builder<Any>()
|
||||||
|
.event(event).id("none").data(it)
|
||||||
|
.build()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -23,6 +23,9 @@ import dev.dnpm.etl.processor.NotFoundException
|
|||||||
import dev.dnpm.etl.processor.monitoring.ReportService
|
import dev.dnpm.etl.processor.monitoring.ReportService
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestId
|
import dev.dnpm.etl.processor.monitoring.RequestId
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
||||||
|
import org.springframework.data.domain.Pageable
|
||||||
|
import org.springframework.data.domain.Sort
|
||||||
|
import org.springframework.data.web.PageableDefault
|
||||||
import org.springframework.stereotype.Controller
|
import org.springframework.stereotype.Controller
|
||||||
import org.springframework.ui.Model
|
import org.springframework.ui.Model
|
||||||
import org.springframework.web.bind.annotation.GetMapping
|
import org.springframework.web.bind.annotation.GetMapping
|
||||||
@ -37,8 +40,24 @@ class HomeController(
|
|||||||
) {
|
) {
|
||||||
|
|
||||||
@GetMapping
|
@GetMapping
|
||||||
fun index(model: Model): String {
|
fun index(
|
||||||
val requests = requestRepository.findAll().sortedByDescending { it.processedAt }.take(25)
|
@PageableDefault(page = 0, size = 20, sort = ["processedAt"], direction = Sort.Direction.DESC) pageable: Pageable,
|
||||||
|
model: Model
|
||||||
|
): String {
|
||||||
|
val requests = requestRepository.findAll(pageable)
|
||||||
|
model.addAttribute("requests", requests)
|
||||||
|
|
||||||
|
return "index"
|
||||||
|
}
|
||||||
|
|
||||||
|
@GetMapping(path = ["patient/{patientId}"])
|
||||||
|
fun byPatient(
|
||||||
|
@PathVariable patientId: String,
|
||||||
|
@PageableDefault(page = 0, size = 20, sort = ["processedAt"], direction = Sort.Direction.DESC) pageable: Pageable,
|
||||||
|
model: Model
|
||||||
|
): String {
|
||||||
|
val requests = requestRepository.findRequestByPatientId(patientId, pageable)
|
||||||
|
model.addAttribute("patientId", patientId)
|
||||||
model.addAttribute("requests", requests)
|
model.addAttribute("requests", requests)
|
||||||
|
|
||||||
return "index"
|
return "index"
|
||||||
|
@ -0,0 +1,47 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.web
|
||||||
|
|
||||||
|
import dev.dnpm.etl.processor.config.SecurityConfigProperties
|
||||||
|
import org.springframework.boot.autoconfigure.security.oauth2.client.OAuth2ClientProperties
|
||||||
|
import org.springframework.stereotype.Controller
|
||||||
|
import org.springframework.ui.Model
|
||||||
|
import org.springframework.web.bind.annotation.GetMapping
|
||||||
|
|
||||||
|
@Controller
|
||||||
|
class LoginController(
|
||||||
|
private val securityConfigProperties: SecurityConfigProperties?,
|
||||||
|
private val oAuth2ClientProperties: OAuth2ClientProperties?
|
||||||
|
) {
|
||||||
|
|
||||||
|
@GetMapping(path = ["/login"])
|
||||||
|
fun login(model: Model): String {
|
||||||
|
if (securityConfigProperties?.enableOidc == true) {
|
||||||
|
model.addAttribute(
|
||||||
|
"oidcLogins",
|
||||||
|
oAuth2ClientProperties?.registration?.map { (key, value) -> Pair(key, value.clientName) }.orEmpty()
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
model.addAttribute("oidcLogins", emptyList<Pair<String, String>>())
|
||||||
|
}
|
||||||
|
return "login"
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -22,6 +22,7 @@ package dev.dnpm.etl.processor.web
|
|||||||
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestType
|
import dev.dnpm.etl.processor.monitoring.RequestType
|
||||||
|
import org.springframework.beans.factory.annotation.Qualifier
|
||||||
import org.springframework.http.MediaType
|
import org.springframework.http.MediaType
|
||||||
import org.springframework.http.codec.ServerSentEvent
|
import org.springframework.http.codec.ServerSentEvent
|
||||||
import org.springframework.web.bind.annotation.GetMapping
|
import org.springframework.web.bind.annotation.GetMapping
|
||||||
@ -38,6 +39,7 @@ import java.time.temporal.ChronoUnit
|
|||||||
@RestController
|
@RestController
|
||||||
@RequestMapping(path = ["/statistics"])
|
@RequestMapping(path = ["/statistics"])
|
||||||
class StatisticsRestController(
|
class StatisticsRestController(
|
||||||
|
@Qualifier("statisticsUpdateProducer")
|
||||||
private val statisticsUpdateProducer: Sinks.Many<Any>,
|
private val statisticsUpdateProducer: Sinks.Many<Any>,
|
||||||
private val requestRepository: RequestRepository
|
private val requestRepository: RequestRepository
|
||||||
) {
|
) {
|
||||||
@ -83,9 +85,9 @@ class StatisticsRestController(
|
|||||||
.groupBy { formatter.format(it.processedAt) }
|
.groupBy { formatter.format(it.processedAt) }
|
||||||
.map {
|
.map {
|
||||||
val requestList = it.value
|
val requestList = it.value
|
||||||
.groupBy { it.status }
|
.groupBy { request -> request.status }
|
||||||
.map {
|
.map { request ->
|
||||||
Pair(it.key, it.value.size)
|
Pair(request.key, request.value.size)
|
||||||
}
|
}
|
||||||
.toMap()
|
.toMap()
|
||||||
Pair(
|
Pair(
|
||||||
@ -152,6 +154,10 @@ class StatisticsRestController(
|
|||||||
.build(),
|
.build(),
|
||||||
ServerSentEvent.builder<Any>()
|
ServerSentEvent.builder<Any>()
|
||||||
.event("deleterequestpatientstates").id("none").data(this.requestPatientStates(true))
|
.event("deleterequestpatientstates").id("none").data(this.requestPatientStates(true))
|
||||||
|
.build(),
|
||||||
|
|
||||||
|
ServerSentEvent.builder<Any>()
|
||||||
|
.event("newrequest").id("none").data("newrequest")
|
||||||
.build()
|
.build()
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
@ -4,12 +4,15 @@ spring:
|
|||||||
file: ./dev-compose.yml
|
file: ./dev-compose.yml
|
||||||
|
|
||||||
app:
|
app:
|
||||||
rest:
|
#rest:
|
||||||
uri: http://localhost:9000/bwhc/etl/api
|
# uri: http://localhost:9000/bwhc/etl/api
|
||||||
kafka:
|
kafka:
|
||||||
topic: test
|
topic: test
|
||||||
response-topic: test-response
|
response-topic: test_response
|
||||||
servers: kafka:9092
|
servers: localhost:9094
|
||||||
|
#security:
|
||||||
|
# admin-user: admin
|
||||||
|
# admin-password: "{noop}very-secret"
|
||||||
|
|
||||||
server:
|
server:
|
||||||
port: 8000
|
port: 8000
|
||||||
|
@ -4,4 +4,18 @@ spring:
|
|||||||
consumer:
|
consumer:
|
||||||
group-id: ${app.kafka.group-id}
|
group-id: ${app.kafka.group-id}
|
||||||
flyway:
|
flyway:
|
||||||
locations: "classpath:db/migration/{vendor}"
|
locations: "classpath:db/migration/{vendor}"
|
||||||
|
|
||||||
|
web:
|
||||||
|
resources:
|
||||||
|
cache:
|
||||||
|
cachecontrol:
|
||||||
|
max-age: 1d
|
||||||
|
chain:
|
||||||
|
strategy:
|
||||||
|
content:
|
||||||
|
enabled: true
|
||||||
|
paths: /**/*.js,/**/*.css,/**/*.svg,/**/*.jpeg
|
||||||
|
|
||||||
|
server:
|
||||||
|
forward-headers-strategy: framework
|
@ -0,0 +1,8 @@
|
|||||||
|
CREATE TABLE IF NOT EXISTS token
|
||||||
|
(
|
||||||
|
id int auto_increment primary key,
|
||||||
|
name varchar(255) not null,
|
||||||
|
username varchar(255) not null unique,
|
||||||
|
password varchar(255) not null,
|
||||||
|
created_at datetime default utc_timestamp() not null
|
||||||
|
);
|
@ -0,0 +1,7 @@
|
|||||||
|
CREATE TABLE IF NOT EXISTS user_role
|
||||||
|
(
|
||||||
|
id int auto_increment primary key,
|
||||||
|
username varchar(255) not null unique,
|
||||||
|
role varchar(255) not null,
|
||||||
|
created_at datetime default utc_timestamp() not null
|
||||||
|
);
|
@ -0,0 +1,9 @@
|
|||||||
|
CREATE TABLE IF NOT EXISTS token
|
||||||
|
(
|
||||||
|
id serial,
|
||||||
|
name varchar(255) not null,
|
||||||
|
username varchar(255) not null unique,
|
||||||
|
password varchar(255) not null,
|
||||||
|
created_at timestamp with time zone default now() not null,
|
||||||
|
PRIMARY KEY (id)
|
||||||
|
);
|
@ -0,0 +1,8 @@
|
|||||||
|
CREATE TABLE IF NOT EXISTS user_role
|
||||||
|
(
|
||||||
|
id serial,
|
||||||
|
username varchar(255) not null unique,
|
||||||
|
role varchar(255) not null,
|
||||||
|
created_at timestamp with time zone default now() not null,
|
||||||
|
PRIMARY KEY (id)
|
||||||
|
);
|
BIN
src/main/resources/static/bg.jpeg
Normal file
After Width: | Height: | Size: 3.0 KiB |
45
src/main/resources/static/echarts.min.js
vendored
41
src/main/resources/static/icon.svg
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||||
|
|
||||||
|
<svg
|
||||||
|
width="256"
|
||||||
|
height="256"
|
||||||
|
viewBox="0 0 67.733332 67.733335"
|
||||||
|
version="1.1"
|
||||||
|
id="svg5"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg">
|
||||||
|
<defs
|
||||||
|
id="defs2" />
|
||||||
|
<g
|
||||||
|
id="layer1">
|
||||||
|
<g
|
||||||
|
id="g26002"
|
||||||
|
transform="matrix(1.5,0,0,1.5,-16.933333,-1.8487648)">
|
||||||
|
<path
|
||||||
|
id="path12437"
|
||||||
|
transform="matrix(0.21771408,0,0,0.21771408,73.025692,24.874779)"
|
||||||
|
style="fill:#f59e00;fill-opacity:1"
|
||||||
|
d="m -110.41995,43.223174 -55.55561,-2e-6 -27.7778,-48.1125685 27.77781,-48.1125655 55.5556,3e-6 27.777803,48.1125679 z" />
|
||||||
|
<path
|
||||||
|
id="path13446"
|
||||||
|
transform="matrix(0.21771408,0,0,0.21771408,54.882836,14.399994)"
|
||||||
|
style="fill:#004d6e;fill-opacity:1"
|
||||||
|
d="m -110.41995,43.223174 -55.55561,-2e-6 -27.7778,-48.1125685 27.77781,-48.1125655 55.5556,3e-6 27.777803,48.1125679 z" />
|
||||||
|
<path
|
||||||
|
id="path13448"
|
||||||
|
transform="matrix(0.21771408,0,0,0.21771408,54.882835,35.349561)"
|
||||||
|
style="fill:#706f6f;fill-opacity:1"
|
||||||
|
d="m -110.41995,43.223174 -55.55561,-2e-6 -27.7778,-48.1125685 27.77781,-48.1125655 55.5556,3e-6 27.777803,48.1125679 z" />
|
||||||
|
<path
|
||||||
|
id="path25844"
|
||||||
|
transform="matrix(0.21771408,0,0,0.21771408,60.930454,24.874778)"
|
||||||
|
style="fill:#ffffff;fill-opacity:1"
|
||||||
|
d="m -110.41995,43.223174 -55.55561,-2e-6 -27.7778,-48.1125685 27.77781,-48.1125655 55.5556,3e-6 27.777803,48.1125679 z" />
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 1.6 KiB |
BIN
src/main/resources/static/kafka.png
Normal file
After Width: | Height: | Size: 1.6 KiB |
@ -4,7 +4,7 @@ const dateFormat = new Intl.DateTimeFormat('de-DE', dateFormatOptions);
|
|||||||
const dateTimeFormatOptions = { year: 'numeric', month: '2-digit', day: '2-digit', hour: '2-digit', minute: 'numeric', second: 'numeric' };
|
const dateTimeFormatOptions = { year: 'numeric', month: '2-digit', day: '2-digit', hour: '2-digit', minute: 'numeric', second: 'numeric' };
|
||||||
const dateTimeFormat = new Intl.DateTimeFormat('de-DE', dateTimeFormatOptions);
|
const dateTimeFormat = new Intl.DateTimeFormat('de-DE', dateTimeFormatOptions);
|
||||||
|
|
||||||
window.onload = () => {
|
const formatTimeElements = () => {
|
||||||
Array.from(document.getElementsByTagName('time')).forEach((timeTag) => {
|
Array.from(document.getElementsByTagName('time')).forEach((timeTag) => {
|
||||||
let date = Date.parse(timeTag.getAttribute('datetime'));
|
let date = Date.parse(timeTag.getAttribute('datetime'));
|
||||||
if (! isNaN(date)) {
|
if (! isNaN(date)) {
|
||||||
@ -13,6 +13,9 @@ window.onload = () => {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
window.addEventListener('load', formatTimeElements);
|
||||||
|
window.addEventListener('htmx:afterRequest', formatTimeElements);
|
||||||
|
|
||||||
function drawPieChart(url, elemId, title, data) {
|
function drawPieChart(url, elemId, title, data) {
|
||||||
if (data) {
|
if (data) {
|
||||||
update(elemId, data);
|
update(elemId, data);
|
||||||
|
BIN
src/main/resources/static/server.png
Normal file
After Width: | Height: | Size: 1.4 KiB |
@ -1,44 +1,153 @@
|
|||||||
|
:root {
|
||||||
|
--text: #333;
|
||||||
|
--table-border: rgba(16, 24, 40, .1);
|
||||||
|
|
||||||
|
--dark: brightness(.90);
|
||||||
|
|
||||||
|
--bg-blue: rgb(0, 74, 157);
|
||||||
|
--bg-blue-op: rgba(0, 74, 157, .35);
|
||||||
|
|
||||||
|
--bg-green: rgb(0, 128, 0);
|
||||||
|
--bg-green-op: rgba(0, 128, 0, .35);
|
||||||
|
|
||||||
|
|
||||||
|
--bg-yellow: rgb(255, 140, 0);
|
||||||
|
--bg-yellow-op: rgba(255, 140, 0, .35);
|
||||||
|
|
||||||
|
|
||||||
|
--bg-red: rgb(255, 0, 0);
|
||||||
|
--bg-red-op: rgba(255, 0, 0, .35);
|
||||||
|
|
||||||
|
--bg-gray: rgb(112, 128, 144);
|
||||||
|
--bg-gray-op: rgba(112, 128, 144, .35);
|
||||||
|
}
|
||||||
|
|
||||||
|
html {
|
||||||
|
background: linear-gradient(-5deg, var(--bg-blue-op), transparent 10em);
|
||||||
|
min-height: 100vh;
|
||||||
|
overflow-y: scroll;
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
margin: 0;
|
margin: 0 0 5em 0;
|
||||||
font-family: sans-serif;
|
font-family: sans-serif;
|
||||||
font-size: .8rem;
|
font-size: .8rem;
|
||||||
color: #333;
|
color: var(--text);
|
||||||
|
|
||||||
|
min-height: 100vh;
|
||||||
|
|
||||||
|
background: url(bg.jpeg) no-repeat;
|
||||||
|
background-size: contain;
|
||||||
}
|
}
|
||||||
|
|
||||||
nav {
|
nav {
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
background: #d5dad5;
|
padding: 1em 0;
|
||||||
height: 3rem;
|
|
||||||
|
line-height: 1.5rem;
|
||||||
max-width: 1140px;
|
max-width: 1140px;
|
||||||
|
|
||||||
|
border-bottom: 1px solid var(--table-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
nav a {
|
nav a.nav-home {
|
||||||
color: #004a8f;
|
float: left;
|
||||||
text-transform: uppercase;
|
|
||||||
|
color: var(--text);
|
||||||
|
line-height: 1.5em;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
line-height: 2rem;
|
|
||||||
font-weight: 700;
|
font-size: 2em;
|
||||||
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
nav a:hover {
|
nav a.nav-home > img {
|
||||||
text-decoration: underline;
|
width: 1.5em;
|
||||||
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
nav > ul {
|
nav > ul {
|
||||||
margin: 0 3rem;
|
margin: 0 0 0 auto;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
|
|
||||||
|
width: max-content;
|
||||||
}
|
}
|
||||||
|
|
||||||
nav > ul > li {
|
nav > ul > li {
|
||||||
background: #fbfbfb;
|
display: inline-block;
|
||||||
display: block;
|
padding: 0 1rem;
|
||||||
float: left;
|
|
||||||
padding: 2px 1rem;
|
|
||||||
border-left: 1px solid #d5dad5;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
nav > ul > li:first-of-type {
|
nav > ul > li.login {
|
||||||
border-left: none;
|
margin: 0 0 0 1em;
|
||||||
|
padding: 0 0 0 2em;
|
||||||
|
border-left: 1px solid var(--table-border);
|
||||||
|
line-height: 3.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > ul > li.login a {
|
||||||
|
text-decoration: none;
|
||||||
|
text-transform: none;
|
||||||
|
padding: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav .login .user-name {
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > ul > li.login > span {
|
||||||
|
display: inline-block;
|
||||||
|
margin: 0 .5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > ul > li.login .user-icon {
|
||||||
|
flex-direction: column;
|
||||||
|
display: inline flex;
|
||||||
|
vertical-align: middle;
|
||||||
|
inline-size: 4em;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > ul > li.login .user-icon img {
|
||||||
|
margin: 0 0 -1em 0;
|
||||||
|
width: 80%;
|
||||||
|
align-self: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > ul > li.login .user-icon span {
|
||||||
|
padding: 0 .6em;
|
||||||
|
color: white;
|
||||||
|
font-size: .8em;
|
||||||
|
font-weight: bold;
|
||||||
|
border-radius: 4px;
|
||||||
|
line-height: normal;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > ul > li.login .user-icon span.guest {
|
||||||
|
background: darkslategray;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > ul > li.login .user-icon span.user {
|
||||||
|
background: darkgreen;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav > ul > li.login .user-icon span.admin {
|
||||||
|
background: darkred;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav li a {
|
||||||
|
color: var(--bg-blue);
|
||||||
|
text-transform: uppercase;
|
||||||
|
text-decoration: none;
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav li a:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: var(--bg-blue);
|
||||||
}
|
}
|
||||||
|
|
||||||
.breadcrumps {
|
.breadcrumps {
|
||||||
@ -57,22 +166,30 @@ nav > ul > li:first-of-type {
|
|||||||
display: inline;
|
display: inline;
|
||||||
}
|
}
|
||||||
|
|
||||||
.breadcrumps ul li+li:before {
|
.breadcrumps ul li + li:before {
|
||||||
padding: .4rem;
|
padding: .4rem;
|
||||||
color: gray;
|
color: gray;
|
||||||
content: "/\00a0";
|
content: "/\00a0";
|
||||||
}
|
}
|
||||||
|
|
||||||
.breadcrumps ul li a {
|
.breadcrumps ul li a {
|
||||||
color: #333333;
|
color: var(--text);
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.centered {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
main {
|
main {
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
max-width: 1140px;
|
max-width: 1140px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
section {
|
||||||
|
margin: 3em 0;
|
||||||
|
}
|
||||||
|
|
||||||
form {
|
form {
|
||||||
margin: 1rem 0;
|
margin: 1rem 0;
|
||||||
padding: 1rem;
|
padding: 1rem;
|
||||||
@ -114,16 +231,139 @@ form.samplecode-input input:focus-visible {
|
|||||||
background: lightgreen;
|
background: lightgreen;
|
||||||
}
|
}
|
||||||
|
|
||||||
table {
|
.login-form {
|
||||||
border-top: 1px solid lightgray;
|
width: fit-content;
|
||||||
border-left: 1px solid lightgray;
|
margin: 3em auto;
|
||||||
border-spacing: 0;
|
padding: 2em 5em;
|
||||||
border-radius: 3px;
|
|
||||||
|
|
||||||
|
border: 1px solid var(--table-border);
|
||||||
|
border-radius: .5em;
|
||||||
|
background: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.login-form form {
|
||||||
|
width: 20em;
|
||||||
|
margin: 0 auto;
|
||||||
|
display: grid;
|
||||||
|
grid-gap: .5em;
|
||||||
|
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.login-form img {
|
||||||
|
margin: 0 auto;
|
||||||
|
width: 4em;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.userrole-form {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.userrole-form form {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
|
||||||
|
border: none;
|
||||||
|
border-radius: 0;
|
||||||
|
background: none;
|
||||||
|
|
||||||
|
text-align: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
.login-form form *,
|
||||||
|
.token-form form * {
|
||||||
|
padding: 0.5em;
|
||||||
|
border: 1px solid var(--table-border);
|
||||||
|
border-radius: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.login-form form hr,
|
||||||
|
.token-form form hr,
|
||||||
|
.userrole-form form hr {
|
||||||
|
padding: 0;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.login-form button,
|
||||||
|
.login-form a.btn,
|
||||||
|
.token-form button {
|
||||||
|
margin: 1em 0;
|
||||||
|
background: var(--bg-blue);
|
||||||
|
color: white;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.userrole-form form select {
|
||||||
|
padding: 0.5em;
|
||||||
|
border: none;
|
||||||
|
border-radius: 3px;
|
||||||
|
line-height: 1.2rem;
|
||||||
|
font-size: 0.8rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.border {
|
||||||
|
padding: 1.5em;
|
||||||
|
border: 1px solid var(--table-border);
|
||||||
|
border-radius: .5em;
|
||||||
|
background: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
table, .chart {
|
||||||
|
border: 1px solid var(--table-border);
|
||||||
|
padding: 1.5em;
|
||||||
|
|
||||||
|
border-spacing: 0;
|
||||||
|
border-radius: .5em;
|
||||||
|
|
||||||
|
background: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
min-width: 100%;
|
min-width: 100%;
|
||||||
font-family: sans-serif;
|
font-family: sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
table.config-table td:first-child {
|
||||||
|
width: 24em;
|
||||||
|
min-width: fit-content;
|
||||||
|
}
|
||||||
|
|
||||||
|
table.config-table td > button:last-of-type {
|
||||||
|
float: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
.border > table {
|
||||||
|
padding: 0;
|
||||||
|
border: none;
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-control {
|
||||||
|
border-radius: .5em;
|
||||||
|
padding: 1em 2em;
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
|
line-height: 1.75em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-control a {
|
||||||
|
padding: 0 .25em;
|
||||||
|
font-size: 1.75em;
|
||||||
|
color: var(--bg-gray);
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-control a[href] {
|
||||||
|
color: var(--bg-blue);
|
||||||
|
}
|
||||||
|
|
||||||
|
.page-control span {
|
||||||
|
padding: 0 .5em;
|
||||||
|
vertical-align: text-bottom;
|
||||||
|
}
|
||||||
|
|
||||||
#samples-table.max {
|
#samples-table.max {
|
||||||
width: 100vw;
|
width: 100vw;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
@ -140,43 +380,97 @@ table.samples {
|
|||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
th {
|
th, td {
|
||||||
background: #eee;
|
padding: 0.4rem .2rem;
|
||||||
}
|
|
||||||
|
|
||||||
td, th {
|
line-height: 2em;
|
||||||
padding: .2rem;
|
|
||||||
|
|
||||||
border-right: 1px solid lightgray;
|
|
||||||
border-bottom: 1px solid lightgray;
|
|
||||||
|
|
||||||
text-align: left;
|
text-align: left;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
vertical-align: top;
|
vertical-align: top;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
th {
|
||||||
|
border-bottom: 1px solid var(--bg-gray);
|
||||||
|
}
|
||||||
|
|
||||||
td {
|
td {
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
|
border-bottom: 1px solid var(--bg-gray-op);
|
||||||
}
|
}
|
||||||
|
|
||||||
td.bg-green, th.bg-green {
|
tr:last-of-type > td {
|
||||||
background: green;
|
border-bottom: none;
|
||||||
color: white;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
td.bg-yellow, th.bg-yellow {
|
td > small {
|
||||||
background: darkorange;
|
display: block;
|
||||||
color: white;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
td.bg-red, th.bg-red {
|
td.patient-id {
|
||||||
background: red;
|
width: 32em;
|
||||||
color: white;
|
text-overflow: ellipsis;
|
||||||
|
overflow: hidden;
|
||||||
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
td.bg-gray, th.bg-gray {
|
td.bg-blue, th.bg-blue,
|
||||||
background: slategray;
|
td.bg-green, th.bg-green,
|
||||||
|
td.bg-yellow, th.bg-yellow,
|
||||||
|
td.bg-red, th.bg-red,
|
||||||
|
td.bg-gray, th.bg-gray
|
||||||
|
{
|
||||||
|
width: 8em;
|
||||||
|
}
|
||||||
|
|
||||||
|
td.bg-blue > small, th.bg-blue > small {
|
||||||
|
background: var(--bg-blue);
|
||||||
color: white;
|
color: white;
|
||||||
|
border-radius: 0.4em;
|
||||||
|
}
|
||||||
|
|
||||||
|
td.bg-green > small, th.bg-green > small {
|
||||||
|
background: var(--bg-green);
|
||||||
|
color: white;
|
||||||
|
border-radius: 0.4em;
|
||||||
|
}
|
||||||
|
|
||||||
|
td.bg-yellow > small, th.bg-yellow > small {
|
||||||
|
background: var(--bg-yellow);
|
||||||
|
color: white;
|
||||||
|
border-radius: 0.4em;
|
||||||
|
}
|
||||||
|
|
||||||
|
td.bg-red > small, th.bg-red > small {
|
||||||
|
background: var(--bg-red);
|
||||||
|
color: white;
|
||||||
|
border-radius: 0.4em;
|
||||||
|
}
|
||||||
|
|
||||||
|
td.bg-gray > small, th.bg-gray > small {
|
||||||
|
background: var(--bg-gray);
|
||||||
|
color: white;
|
||||||
|
border-radius: 0.4em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg-path {
|
||||||
|
background: var(--bg-gray-op);
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg-from {
|
||||||
|
background: var(--bg-red-op);
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg-to {
|
||||||
|
background: var(--bg-green-op);
|
||||||
|
}
|
||||||
|
|
||||||
|
.bg-path, .bg-from, .bg-to {
|
||||||
|
padding: 0.25rem 0.5rem;
|
||||||
|
border-radius: 3px;
|
||||||
|
|
||||||
|
font-family: monospace;
|
||||||
}
|
}
|
||||||
|
|
||||||
td.bg-shaded, th.bg-shaded {
|
td.bg-shaded, th.bg-shaded {
|
||||||
@ -196,7 +490,6 @@ td.clipboard.clipped {
|
|||||||
padding: 4px 8px;
|
padding: 4px 8px;
|
||||||
|
|
||||||
line-height: 1.2rem;
|
line-height: 1.2rem;
|
||||||
vertical-align: middle;
|
|
||||||
|
|
||||||
border: 0 solid transparent;
|
border: 0 solid transparent;
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
||||||
@ -208,38 +501,38 @@ td.clipboard.clipped {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.btn:active,
|
||||||
.btn:hover {
|
.btn:hover {
|
||||||
filter: drop-shadow(1px 2px 2px gray);
|
filter: drop-shadow(0px 1px 1px var(--bg-gray)) var(--dark);
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn:active {
|
.btn:active {
|
||||||
filter: drop-shadow(1px 1px 2px gray);
|
|
||||||
translate: 0 1px;
|
translate: 0 1px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn.btn-red {
|
.btn.btn-red {
|
||||||
background: red;
|
background: var(--bg-red);
|
||||||
color: white;
|
color: white;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn.btn-red:hover, .btn.btn-red:active {
|
|
||||||
background: darkred !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.btn.btn-blue {
|
.btn.btn-blue {
|
||||||
background: slategray;
|
background: var(--bg-blue);
|
||||||
color: white;
|
color: white;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn.btn-blue:hover, .btn.btn-blue:active {
|
|
||||||
background: darkslategray !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.btn.btn-delete:before {
|
.btn.btn-delete:before {
|
||||||
content: '\1F5D1';
|
content: '\1F5D1';
|
||||||
padding: .2rem;
|
padding: .2rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
button:disabled,
|
||||||
|
.btn:disabled {
|
||||||
|
background: slategray !important;
|
||||||
|
color: lightgray;
|
||||||
|
filter: none;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
|
||||||
input.inline {
|
input.inline {
|
||||||
border: none;
|
border: none;
|
||||||
font-size: 1.1rem;
|
font-size: 1.1rem;
|
||||||
@ -275,19 +568,124 @@ input.inline:focus-visible {
|
|||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
.chart {
|
.charts {
|
||||||
padding: 1rem;
|
display: grid;
|
||||||
margin: .2rem;
|
grid-gap: 1em;
|
||||||
|
grid-template:
|
||||||
|
"a b" 28em
|
||||||
|
"c c" 28em / 1fr 1fr;
|
||||||
|
}
|
||||||
|
|
||||||
border: 1px solid lightgray;
|
.charts > .grid-left {
|
||||||
|
grid-area: a;
|
||||||
|
}
|
||||||
|
|
||||||
|
.charts > .grid-right {
|
||||||
|
grid-area: b;
|
||||||
|
}
|
||||||
|
|
||||||
|
.charts > .grid-full {
|
||||||
|
grid-area: c;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-display {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 10em 16em 10em;
|
||||||
|
place-items: center;
|
||||||
|
width: fit-content;
|
||||||
|
margin: 1em auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-display > * {
|
||||||
|
text-align: center;
|
||||||
|
margin: auto 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-display .connection {
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
height: 4px;
|
||||||
|
background: repeating-linear-gradient(to left, white, white 2px, transparent 2px, transparent 8px, white 8px) var(--bg-red);
|
||||||
|
}
|
||||||
|
|
||||||
|
.connection-display .connection.available {
|
||||||
|
background: var(--bg-green);
|
||||||
|
}
|
||||||
|
|
||||||
|
.notification {
|
||||||
|
margin: 1em;
|
||||||
|
padding: .5em;
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
width: calc(100% - 2.4rem - 4px);
|
.notification.success {
|
||||||
height: 320px;
|
color: var(--bg-green);
|
||||||
|
}
|
||||||
|
|
||||||
|
.notification.notice {
|
||||||
|
color: var(--bg-yellow);
|
||||||
|
}
|
||||||
|
|
||||||
|
.notification.error {
|
||||||
|
color: var(--bg-red);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab {
|
||||||
|
padding: 1em;
|
||||||
|
border: none;
|
||||||
|
border-radius: 3px 3px 0 0;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s;
|
||||||
|
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab:hover,
|
||||||
|
.tab.active {
|
||||||
|
background: var(--table-border);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabcontent {
|
||||||
|
border: 1px solid var(--table-border);
|
||||||
|
border-radius: 0 .5em .5em .5em;
|
||||||
|
display: none;
|
||||||
|
padding: 1em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabcontent.active {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
a.reload {
|
||||||
|
display: none;
|
||||||
|
position: absolute;
|
||||||
|
height: 1.2em;
|
||||||
|
width: 1.2em;
|
||||||
|
background: var(--bg-red);
|
||||||
|
border-radius: 50%;
|
||||||
|
|
||||||
|
color: white;
|
||||||
|
text-decoration: none;
|
||||||
|
font-size: .6em;
|
||||||
|
align-content: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-token {
|
||||||
|
padding: 1em;
|
||||||
|
background: var(--bg-green-op);
|
||||||
|
}
|
||||||
|
|
||||||
|
.new-token > pre {
|
||||||
|
margin: 0;
|
||||||
|
border: 1px solid var(--bg-green);
|
||||||
|
padding: .5em;
|
||||||
|
width: max-content;
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.chart-50pc {
|
.no-token {
|
||||||
width: calc(50% - 2.4rem - 4px);
|
padding: 1em;
|
||||||
|
background: var(--bg-red-op);
|
||||||
}
|
}
|
11
src/main/resources/static/user.svg
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||||
|
<svg width="24" height="24" version="1.1" viewBox="0 0 6.35 6.35" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<g transform="matrix(1.2 0 0 1.2 -108.01 -85.977)">
|
||||||
|
<rect x="90.01" y="71.647" width="5.2917" height="5.2917" rx=".96212" fill="#b3b3b3"/>
|
||||||
|
<g transform="matrix(1.6667 0 0 1.6667 -60.888 -47.952)" fill="#fff">
|
||||||
|
<circle cx="92.126" cy="72.802" r=".70556"/>
|
||||||
|
<path d="m91.068 74.598a1.0583 1.0583 0 0 1 1.0583-1.0583 1.0583 1.0583 0 0 1 1.0583 1.0583h-1.0583z"/>
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 588 B |
125
src/main/resources/templates/configs.html
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="de" xmlns:th="http://www.thymeleaf.org">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>ETL-Prozessor</title>
|
||||||
|
<link rel="stylesheet" th:href="@{/style.css}" />
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div th:replace="~{fragments.html :: nav}"></div>
|
||||||
|
<main>
|
||||||
|
<h1>Konfiguration</h1>
|
||||||
|
|
||||||
|
<div class="tabs">
|
||||||
|
<button class="tab active" onclick="selectTab(this, 'common');">Allgemeine Informationen</button>
|
||||||
|
<button class="tab" onclick="selectTab(this, 'security');">Sicherheit</button>
|
||||||
|
<button class="tab" onclick="selectTab(this, 'transformation');">Transformationen</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="common" class="tabcontent active">
|
||||||
|
<section>
|
||||||
|
<h2>🔧 Allgemeine Konfiguration</h2>
|
||||||
|
<table class="config-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Name</th>
|
||||||
|
<th>Wert</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>Pseudonym erzeugt über</td>
|
||||||
|
<td>[[ ${pseudonymGenerator} ]]</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td>MTBFile-Sender</td>
|
||||||
|
<td>[[ ${mtbFileSender} ]]</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td th:if="${mtbFileSender.startsWith('Rest')}">REST-Endpunkt</td>
|
||||||
|
<td th:if="${mtbFileSender.startsWith('Kafka')}">Kafka-Broker und Topics</td>
|
||||||
|
<td>[[ ${mtbFileEndpoint} ]]</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section hx-ext="sse" th:sse-connect="@{/configs/events}">
|
||||||
|
<div th:insert="~{configs/gPasConnectionAvailable.html}" th:hx-get="@{/configs?gPasConnectionAvailable}" hx-trigger="sse:gpas-connection-check">
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section hx-ext="sse" th:sse-connect="@{/configs/events}">
|
||||||
|
<div th:insert="~{configs/outputConnectionAvailable.html}" th:hx-get="@{/configs?outputConnectionAvailable}" hx-trigger="sse:output-connection-check">
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="security" class="tabcontent">
|
||||||
|
<section th:insert="~{configs/tokens.html}">
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section th:insert="~{configs/userroles.html}">
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="transformation" class="tabcontent">
|
||||||
|
<section>
|
||||||
|
<h2><span th:if="${not transformations.isEmpty()}">✅</span><span th:if="${transformations.isEmpty()}">⛔</span> Transformationen</h2>
|
||||||
|
|
||||||
|
<h3>Syntax</h3>
|
||||||
|
Hier einige Beispiele zum Syntax des JSON-Path
|
||||||
|
<ul>
|
||||||
|
<li style="padding: 0.6rem 0;"><span class="bg-path">diagnoses[*].icdO3T.version</span>: Ersetze die ICD-O3T-Version in allen Diagnosen, z.B. zur Version der deutschen Übersetzung</li>
|
||||||
|
<li style="padding: 0.6rem 0;"><span class="bg-path">patient.gender</span>: Ersetze das Geschlecht des Patienten, z.B. in das von bwHC verlangte Format</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<h3>Konfigurierte Transformationen</h3>
|
||||||
|
<th:block th:if="${transformations.isEmpty()}">
|
||||||
|
<p>
|
||||||
|
Keine konfigurierten Transformationen.
|
||||||
|
</p>
|
||||||
|
</th:block>
|
||||||
|
<th:block th:if="${not transformations.isEmpty()}">
|
||||||
|
<p>
|
||||||
|
Hier sehen Sie eine Übersicht der konfigurierten Transformationen.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<table class="config-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>JSON-Path</th>
|
||||||
|
<th>Transformation von ⇒ nach</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr th:each="transformation : ${transformations}">
|
||||||
|
<td>
|
||||||
|
<span class="bg-path" title="Ersetze Wert(e) an dieser Stelle im MTB-File">[[ ${transformation.path} ]]</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<span class="bg-from" title="Ersetze immer dann, wenn dieser Wert enthalten ist">[[ ${transformation.existingValue} ]]</span>
|
||||||
|
<strong>⇒</strong>
|
||||||
|
<span class="bg-to" title="Ersetze durch diesen Wert">[[ ${transformation.newValue} ]]</span>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</th:block>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
<script th:src="@{/scripts.js}"></script>
|
||||||
|
<script th:src="@{/webjars/htmx.org/dist/htmx.min.js}"></script>
|
||||||
|
<script th:src="@{/webjars/htmx.org/dist/ext/sse.js}"></script>
|
||||||
|
<script>
|
||||||
|
function selectTab(self, elem) {
|
||||||
|
Array.from(document.getElementsByClassName('tab')).forEach(e => e.className = 'tab');
|
||||||
|
self.className = 'tab active';
|
||||||
|
|
||||||
|
Array.from(document.getElementsByClassName('tabcontent')).forEach(e => e.className = 'tabcontent');
|
||||||
|
document.getElementById(elem).className = 'tabcontent active';
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
@ -0,0 +1,19 @@
|
|||||||
|
<th:block th:if="${gPasConnectionAvailable == null}">
|
||||||
|
<h2><span>🟦</span> gPAS nicht konfiguriert - Patienten-IDs werden intern anonymisiert</h2>
|
||||||
|
</th:block>
|
||||||
|
<th:block th:if="${gPasConnectionAvailable != null}">
|
||||||
|
<h2><span th:if="${gPasConnectionAvailable}">✅</span><span th:if="${not(gPasConnectionAvailable)}">⚡</span> Verbindung zu gPAS</h2>
|
||||||
|
<div>
|
||||||
|
Die Verbindung ist aktuell
|
||||||
|
<strong th:if="${gPasConnectionAvailable}" style="color: green">verfügbar.</strong>
|
||||||
|
<strong th:if="${not(gPasConnectionAvailable)}" style="color: red">nicht verfügbar.</strong>
|
||||||
|
</div>
|
||||||
|
<div class="connection-display border">
|
||||||
|
<img th:src="@{/server.png}" alt="ETL-Processor" />
|
||||||
|
<span class="connection" th:classappend="${gPasConnectionAvailable ? 'available' : ''}"></span>
|
||||||
|
<img th:src="@{/server.png}" alt="gPAS" />
|
||||||
|
<span>ETL-Processor</span>
|
||||||
|
<span></span>
|
||||||
|
<span>gPAS</span>
|
||||||
|
</div>
|
||||||
|
</th:block>
|
@ -0,0 +1,16 @@
|
|||||||
|
<h2><span th:if="${outputConnectionAvailable}">✅</span><span th:if="${not(outputConnectionAvailable)}">⚡</span> MTB-File Verbindung</h2>
|
||||||
|
<div>
|
||||||
|
Verbindung über <code>[[ ${mtbFileSender} ]]</code>. Die Verbindung ist aktuell
|
||||||
|
<strong th:if="${outputConnectionAvailable}" style="color: green">verfügbar.</strong>
|
||||||
|
<strong th:if="${not(outputConnectionAvailable)}" style="color: red">nicht verfügbar.</strong>
|
||||||
|
</div>
|
||||||
|
<div class="connection-display border">
|
||||||
|
<img th:src="@{/server.png}" alt="ETL-Processor" />
|
||||||
|
<span class="connection" th:classappend="${outputConnectionAvailable ? 'available' : ''}"></span>
|
||||||
|
<img th:if="${mtbFileSender.startsWith('Rest')}" th:src="@{/server.png}" alt="bwHC-Backend" />
|
||||||
|
<img th:if="${mtbFileSender.startsWith('Kafka')}" th:src="@{/kafka.png}" alt="Kafka-Broker" />
|
||||||
|
<span>ETL-Processor</span>
|
||||||
|
<span></span>
|
||||||
|
<span th:if="${mtbFileSender.startsWith('Rest')}">bwHC-Backend</span>
|
||||||
|
<span th:if="${mtbFileSender.startsWith('Kafka')}">Kafka-Broker</span>
|
||||||
|
</div>
|
40
src/main/resources/templates/configs/tokens.html
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
<div th:if="${not tokensEnabled}">
|
||||||
|
<h2><span>⛔</span> Tokens</h2>
|
||||||
|
<p>Die Verwendung von Tokens ist nicht aktiviert.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="tokens" th:if="${tokensEnabled}">
|
||||||
|
<h2><span>✅</span> Tokens</h2>
|
||||||
|
<div class="border">
|
||||||
|
<div th:if="${tokens.isEmpty()}">Noch keine Tokens vorhanden.</div>
|
||||||
|
<table th:if="${not tokens.isEmpty()}" class="config-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Name</th>
|
||||||
|
<th>Erstellt</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr th:each="token : ${tokens}">
|
||||||
|
<td>[[ ${token.name} ]]</td>
|
||||||
|
<td>
|
||||||
|
<time th:datetime="${token.createdAt}">[[ ${token.createdAt} ]]</time>
|
||||||
|
<button class="btn btn-red" th:hx-delete="@{/configs/tokens/{id}(id=${token.id})}" hx-target="#tokens">Löschen</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div th:if="${newTokenValue != null and success}" class="new-token">
|
||||||
|
Verwendung über HTTP-Basic. Bitte notieren, wird nicht erneut angezeigt: <pre>[[ ${newTokenValue} ]]</pre>
|
||||||
|
</div>
|
||||||
|
<div th:if="${success != null and not success}" class="no-token">
|
||||||
|
Das Token konnte nicht erzeugt werden. Versuchen Sie einen anderen Namen.
|
||||||
|
</div>
|
||||||
|
<div class="token-form">
|
||||||
|
<form th:hx-post="@{/configs/tokens}" hx-target="#tokens">
|
||||||
|
<input placeholder="Token-Name" name="name" required />
|
||||||
|
<button class="btn">Token Erstellen</button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
37
src/main/resources/templates/configs/userroles.html
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
<div th:if="${not userRolesEnabled}">
|
||||||
|
<h2><span>⛔</span> Benutzerberechtigungen</h2>
|
||||||
|
<p>Die Verwendung von rollenbasierten Benutzerberechtigungen ist nicht aktiviert.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="userroles" th:if="${userRolesEnabled}">
|
||||||
|
<h2><span>✅</span> Benutzerberechtigungen</h2>
|
||||||
|
<div class="border">
|
||||||
|
<div th:if="${userRoles.isEmpty()}">Noch keine Benutzerberechtigungen vorhanden.</div>
|
||||||
|
<table th:if="${not userRoles.isEmpty()}" class="config-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Benutzername</th>
|
||||||
|
<th>Rolle</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr th:each="userRole : ${userRoles}">
|
||||||
|
<td>[[ ${userRole.username} ]]</td>
|
||||||
|
<td>
|
||||||
|
<div class="userrole-form">
|
||||||
|
<form th:hx-put="@{/configs/userroles/{id}(id=${userRole.id})}" hx-target="#userroles">
|
||||||
|
<select name="role" th:disabled="${#authorization.authentication.getName() == userRole.username}">
|
||||||
|
<option th:selected="${userRole.role.value == 'guest'}" value="GUEST">Gast</option>
|
||||||
|
<option th:selected="${userRole.role.value == 'user'}" value="USER">Benutzer</option>
|
||||||
|
<option th:selected="${userRole.role.value == 'admin'}" value="ADMIN">Administrator</option>
|
||||||
|
</select>
|
||||||
|
<button class="btn btn-blue" th:disabled="${#authorization.authentication.getName() == userRole.username}">Übernehmen</button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
<button class="btn btn-red" th:hx-delete="@{/configs/userroles/{id}(id=${userRole.id})}" hx-target="#userroles" th:disabled="${#authorization.authentication.getName() == userRole.username}">Löschen</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
@ -1,5 +1,5 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html xmlns:th="http://www.thymeleaf.org">
|
<html xmlns:th="http://www.thymeleaf.org" xmlns:sec="http://www.thymeleaf.org">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<link rel="stylesheet" th:href="@{/style.css}" />
|
<link rel="stylesheet" th:href="@{/style.css}" />
|
||||||
@ -7,9 +7,33 @@
|
|||||||
<body>
|
<body>
|
||||||
<div th:fragment="nav">
|
<div th:fragment="nav">
|
||||||
<nav>
|
<nav>
|
||||||
|
<span>
|
||||||
|
<a class="nav-home" th:href="@{/}">
|
||||||
|
<img th:src="@{/icon.svg}" alt="Icon" />
|
||||||
|
<span>ETL-Processor</span>
|
||||||
|
</a>
|
||||||
|
</span>
|
||||||
<ul>
|
<ul>
|
||||||
<li><a th:href="@{/}">Übersicht</a></li>
|
<li><a th:href="@{/}">Übersicht</a></li>
|
||||||
<li><a th:href="@{/statistics}">Statistiken</a></li>
|
<li><a th:href="@{/statistics}">Statistiken</a></li>
|
||||||
|
<li sec:authorize="hasRole('ADMIN')">
|
||||||
|
<a th:href="@{/configs}">Konfiguration</a>
|
||||||
|
</li>
|
||||||
|
<li class="login" sec:authorize="not isAuthenticated()">
|
||||||
|
<a class="btn btn-blue" th:href="@{/login}">Login</a>
|
||||||
|
</li>
|
||||||
|
<li class="login" sec:authorize="isAuthenticated()">
|
||||||
|
<span>
|
||||||
|
<div class="user-icon">
|
||||||
|
<img th:src="@{/user.svg}" alt="User-Image">
|
||||||
|
<span sec:authorize="hasRole('ADMIN')" class="user-role admin">Admin</span>
|
||||||
|
<span sec:authorize="hasRole('USER')" class="user-role user">User</span>
|
||||||
|
<span sec:authorize="hasRole('GUEST')" class="user-role guest">Guest</span>
|
||||||
|
</div>
|
||||||
|
<span class="user-name" sec:authentication="name">?</span>
|
||||||
|
</span>
|
||||||
|
<a class="btn btn-red" th:href="@{/logout}">Abmelden</a>
|
||||||
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</nav>
|
</nav>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="de" xmlns:th="http://www.thymeleaf.org">
|
<html lang="de" xmlns:th="http://www.thymeleaf.org" xmlns:sec="http://www.thymeleaf.org">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<title>ETL-Prozessor</title>
|
<title>ETL-Prozessor</title>
|
||||||
@ -9,37 +9,91 @@
|
|||||||
<div th:replace="~{fragments.html :: nav}"></div>
|
<div th:replace="~{fragments.html :: nav}"></div>
|
||||||
<main>
|
<main>
|
||||||
|
|
||||||
<h1>Letzte Anfragen</h1>
|
<h1>Alle Anfragen<a id="reload-notify" class="reload" title="Neue Anfragen" th:href="@{/}">⟳</a></h1>
|
||||||
|
|
||||||
<table>
|
<div>
|
||||||
<thead>
|
<h2 th:if="${patientId != null}">
|
||||||
<tr>
|
Betreffend Patienten-Pseudonym <span class="monospace" th:text="${patientId}">***</span>
|
||||||
<th>Status</th>
|
<a class="btn btn-blue" th:if="${patientId != null}" th:href="@{/}">Alle anzeigen</a>
|
||||||
<th>Typ</th>
|
</h2>
|
||||||
<th>ID</th>
|
</div>
|
||||||
<th>Datum</th>
|
|
||||||
<th>Patienten-ID</th>
|
<div class="border">
|
||||||
</tr>
|
<div th:if="${patientId == null}" class="page-control">
|
||||||
</thead>
|
<a id="first-page-link" th:href="@{/(page=${0})}" title="Zum Anfang: Taste W" th:if="${not requests.isFirst()}">⇤</a><a th:if="${requests.isFirst()}">⇤</a>
|
||||||
<tbody>
|
<a id="prev-page-link" th:href="@{/(page=${requests.getNumber() - 1})}" title="Seite zurück: Taste A" th:if="${not requests.isFirst()}">←</a><a th:if="${requests.isFirst()}">←</a>
|
||||||
<tr th:each="request : ${requests}">
|
<span>Seite [[ ${requests.getNumber() + 1} ]] von [[ ${requests.getTotalPages()} ]]</span>
|
||||||
<td th:if="${request.status.value.contains('success')}" class="bg-green"><small>[[ ${request.status} ]]</small></td>
|
<a id="next-page-link" th:href="@{/(page=${requests.getNumber() + 1})}" title="Seite vor: Taste D" th:if="${not requests.isLast()}">→</a><a th:if="${requests.isLast()}">→</a>
|
||||||
<td th:if="${request.status.value.contains('warning')}" class="bg-yellow"><small>[[ ${request.status} ]]</small></td>
|
<a id="last-page-link" th:href="@{/(page=${requests.getTotalPages() - 1})}" title="Zum Ende: Taste S" th:if="${not requests.isLast()}">⇥</a><a th:if="${requests.isLast()}">⇥</a>
|
||||||
<td th:if="${request.status.value.contains('error')}" class="bg-red"><small>[[ ${request.status} ]]</small></td>
|
</div>
|
||||||
<td th:if="${request.status.value == 'unknown'}" class="bg-gray"><small>[[ ${request.status} ]]</small></td>
|
<div th:if="${patientId != null}" class="page-control">
|
||||||
<td th:if="${request.status.value == 'duplication'}" class="bg-gray"><small>[[ ${request.status} ]]</small></td>
|
<a id="first-page-link" th:href="@{/patient/{patientId}(patientId=${patientId},page=${0})}" title="Zum Anfang: Taste W" th:if="${not requests.isFirst()}">⇤</a><a th:if="${requests.isFirst()}">⇤</a>
|
||||||
<td th:style="${request.type.value == 'delete'} ? 'color: red;'"><small>[[ ${request.type} ]]</small></td>
|
<a id="prev-page-link" th:href="@{/patient/{patientId}(patientId=${patientId},page=${requests.getNumber() - 1})}" title="Seite zurück: Taste A" th:if="${not requests.isFirst()}">←</a><a th:if="${requests.isFirst()}">←</a>
|
||||||
<td th:if="not ${request.report}">[[ ${request.uuid} ]]</td>
|
<span>Seite [[ ${requests.getNumber() + 1} ]] von [[ ${requests.getTotalPages()} ]]</span>
|
||||||
<td th:if="${request.report}">
|
<a id="next-page-link" th:href="@{/patient/{patientId}(patientId=${patientId},page=${requests.getNumber() + 1})}" title="Seite vor: Taste D" th:if="${not requests.isLast()}">→</a><a th:if="${requests.isLast()}">→</a>
|
||||||
<a th:href="@{/report/{id}(id=${request.uuid})}">[[ ${request.uuid} ]]</a>
|
<a id="last-page-link" th:href="@{/patient/{patientId}(patientId=${patientId},page=${requests.getTotalPages() - 1})}" title="Zum Ende: Taste S" th:if="${not requests.isLast()}">⇥</a><a th:if="${requests.isLast()}">⇥</a>
|
||||||
</td>
|
</div>
|
||||||
<td><time th:datetime="${request.processedAt}">[[ ${request.processedAt} ]]</time></td>
|
<table class="paged">
|
||||||
<td>[[ ${request.patientId} ]]</td>
|
<thead>
|
||||||
</tr>
|
<tr>
|
||||||
</tbody>
|
<th>Status</th>
|
||||||
</table>
|
<th>Typ</th>
|
||||||
|
<th>ID</th>
|
||||||
|
<th>Datum</th>
|
||||||
|
<th>Patienten-ID</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<tr th:each="request : ${requests}">
|
||||||
|
<td th:if="${request.status.value.contains('success')}" class="bg-green"><small>[[ ${request.status} ]]</small></td>
|
||||||
|
<td th:if="${request.status.value.contains('warning')}" class="bg-yellow"><small>[[ ${request.status} ]]</small></td>
|
||||||
|
<td th:if="${request.status.value.contains('error')}" class="bg-red"><small>[[ ${request.status} ]]</small></td>
|
||||||
|
<td th:if="${request.status.value == 'unknown'}" class="bg-gray"><small>[[ ${request.status} ]]</small></td>
|
||||||
|
<td th:if="${request.status.value == 'duplication'}" class="bg-gray"><small>[[ ${request.status} ]]</small></td>
|
||||||
|
<td th:style="${request.type.value == 'delete'} ? 'color: red;'"><small>[[ ${request.type} ]]</small></td>
|
||||||
|
<td th:if="not ${request.report}">[[ ${request.uuid} ]]</td>
|
||||||
|
<td th:if="${request.report}">
|
||||||
|
<a th:href="@{/report/{id}(id=${request.uuid})}" sec:authorize="hasRole('USER') or hasRole('ADMIN')">[[ ${request.uuid} ]]</a>
|
||||||
|
<th:block sec:authorize="not (hasRole('USER') or hasRole('ADMIN'))">[[ ${request.uuid} ]]</th:block>
|
||||||
|
</td>
|
||||||
|
<td><time th:datetime="${request.processedAt}">[[ ${request.processedAt} ]]</time></td>
|
||||||
|
<td class="patient-id" th:if="${patientId != null}" sec:authorize="hasRole('USER') or hasRole('ADMIN')">
|
||||||
|
[[ ${request.patientId} ]]
|
||||||
|
</td>
|
||||||
|
<td class="patient-id" th:if="${patientId == null}" sec:authorize="hasRole('USER') or hasRole('ADMIN')">
|
||||||
|
<a th:href="@{/patient/{pid}(pid=${request.patientId})}">[[ ${request.patientId} ]]</a>
|
||||||
|
</td>
|
||||||
|
<td class="patient-id" sec:authorize="not (hasRole('USER') or hasRole('ADMIN'))">***</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
|
||||||
</main>
|
</main>
|
||||||
<script th:src="@{/scripts.js}"></script>
|
<script th:src="@{/scripts.js}"></script>
|
||||||
|
<script>
|
||||||
|
window.addEventListener('load', () => {
|
||||||
|
let keyBindings = {
|
||||||
|
'w': 'first-page-link',
|
||||||
|
'a': 'prev-page-link',
|
||||||
|
'd': 'next-page-link',
|
||||||
|
's': 'last-page-link'
|
||||||
|
};
|
||||||
|
window.onkeydown = (event) => {
|
||||||
|
for (const [key, elemId] of Object.entries(keyBindings)) {
|
||||||
|
if (event.key === key && document.getElementById(elemId)) {
|
||||||
|
document.getElementById(elemId).style.background = 'yellow';
|
||||||
|
document.getElementById(elemId).click();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
const eventSource = new EventSource('statistics/events');
|
||||||
|
eventSource.addEventListener('newrequest', event => {
|
||||||
|
console.log(event);
|
||||||
|
document.getElementById('reload-notify').style.display = 'inline-flex';
|
||||||
|
});
|
||||||
|
</script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
27
src/main/resources/templates/login.html
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="de" xmlns:th="http://www.thymeleaf.org">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>ETL-Prozessor</title>
|
||||||
|
<link rel="stylesheet" th:href="@{/style.css}" />
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div th:replace="~{fragments.html :: nav}"></div>
|
||||||
|
<main>
|
||||||
|
<div class="login-form">
|
||||||
|
<img th:src="@{/user.svg}" alt="user-logo" />
|
||||||
|
<h2 class="centered">Anmelden</h2>
|
||||||
|
<div class="centered notification error" th:if="${param.error}">Anmeldung nicht erfolgreich</div>
|
||||||
|
<div class="centered notification notice" th:if="${param.expired}">Sitzung abgelaufen oder von einem Administrator beendet.</div>
|
||||||
|
<div class="centered notification success" th:if="${param.logout}">Sie haben sich abgemeldet</div>
|
||||||
|
<form method="post" th:action="@{/login}">
|
||||||
|
<input type="text" id="username" name="username" class="form-control" placeholder="Username" required="" autofocus="" />
|
||||||
|
<input type="password" id="password" name="password" class="form-control" placeholder="Password" required="" />
|
||||||
|
<button class="btn" type="submit">Anmelden</button>
|
||||||
|
<hr th:if="${not oidcLogins.isEmpty()}" />
|
||||||
|
<a th:each="oidcLogin : ${oidcLogins}" class="btn" th:href="@{/oauth2/authorization/{provider}(provider=${oidcLogin.component1()})}">OIDC Login - [[ ${oidcLogin.component2()} ]]</a>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
</body>
|
||||||
|
</html>
|
@ -1,5 +1,5 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="de" xmlns:th="http://www.thymeleaf.org">
|
<html lang="de" xmlns:th="http://www.thymeleaf.org" xmlns:sec="http://www.thymeleaf.org">
|
||||||
<head>
|
<head>
|
||||||
<meta charset="UTF-8">
|
<meta charset="UTF-8">
|
||||||
<title>ETL-Prozessor</title>
|
<title>ETL-Prozessor</title>
|
||||||
@ -15,6 +15,7 @@
|
|||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>Status</th>
|
<th>Status</th>
|
||||||
|
<th>Typ</th>
|
||||||
<th>ID</th>
|
<th>ID</th>
|
||||||
<th>Datum</th>
|
<th>Datum</th>
|
||||||
<th>Patienten-ID</th>
|
<th>Patienten-ID</th>
|
||||||
@ -27,26 +28,34 @@
|
|||||||
<td th:if="${request.status.value == 'error'}" class="bg-red"><small>[[ ${request.status} ]]</small></td>
|
<td th:if="${request.status.value == 'error'}" class="bg-red"><small>[[ ${request.status} ]]</small></td>
|
||||||
<td th:if="${request.status.value == 'unknown'}" class="bg-gray"><small>[[ ${request.status} ]]</small></td>
|
<td th:if="${request.status.value == 'unknown'}" class="bg-gray"><small>[[ ${request.status} ]]</small></td>
|
||||||
<td th:if="${request.status.value == 'duplication'}" class="bg-gray"><small>[[ ${request.status} ]]</small></td>
|
<td th:if="${request.status.value == 'duplication'}" class="bg-gray"><small>[[ ${request.status} ]]</small></td>
|
||||||
|
<td th:style="${request.type.value == 'delete'} ? 'color: red;'"><small>[[ ${request.type} ]]</small></td>
|
||||||
<td>[[ ${request.uuid} ]]</td>
|
<td>[[ ${request.uuid} ]]</td>
|
||||||
<td><time th:datetime="${request.processedAt}">[[ ${request.processedAt} ]]</time></td>
|
<td><time th:datetime="${request.processedAt}">[[ ${request.processedAt} ]]</time></td>
|
||||||
<td>[[ ${request.patientId} ]]</td>
|
<td class="patient-id" sec:authorize="authenticated">[[ ${request.patientId} ]]</td>
|
||||||
|
<td class="patient-id" sec:authorize="not authenticated">***</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
<h2 th:text="${request.report.description}"></h2>
|
<h2 th:text="${request.report.description}"></h2>
|
||||||
|
|
||||||
<table th:if="not ${issues.isEmpty()}">
|
<p th:if="${issues.isEmpty()}">
|
||||||
|
Keine weiteren Angaben.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<table th:if="${not issues.isEmpty()}">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>Schweregrad</th>
|
<th>Schweregrad</th>
|
||||||
<th>Beschreibung</th>
|
<th>Beschreibung</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
<tr th:each="issue : ${issues}">
|
<tr th:each="issue : ${issues}">
|
||||||
|
<td th:if="${issue.severity.value == 'info'}" class="bg-blue"><small>[[ ${issue.severity} ]]</small></td>
|
||||||
<td th:if="${issue.severity.value == 'warning'}" class="bg-yellow"><small>[[ ${issue.severity} ]]</small></td>
|
<td th:if="${issue.severity.value == 'warning'}" class="bg-yellow"><small>[[ ${issue.severity} ]]</small></td>
|
||||||
<td th:if="${issue.severity.value == 'error'}" class="bg-red"><small>[[ ${issue.severity} ]]</small></td>
|
<td th:if="${issue.severity.value == 'error'}" class="bg-red"><small>[[ ${issue.severity} ]]</small></td>
|
||||||
|
<td th:if="${issue.severity.value == 'fatal'}" class="bg-red"><small>[[ ${issue.severity} ]]</small></td>
|
||||||
<td>[[ ${issue.message} ]]</td>
|
<td>[[ ${issue.message} ]]</td>
|
||||||
</tr>
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
|
@ -13,28 +13,32 @@
|
|||||||
Hier sehen Sie eine Übersicht über eingegangene Anfragen.
|
Hier sehen Sie eine Übersicht über eingegangene Anfragen.
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<h2>MTB-File-Anfragen</h2>
|
<section>
|
||||||
<p>
|
<h2>MTB-File-Anfragen</h2>
|
||||||
Anfragen zur Aktualisierung von Patientendaten durch Übermittlung eines MTB-Files.
|
<p>
|
||||||
</p>
|
Anfragen zur Aktualisierung von Patientendaten durch Übermittlung eines MTB-Files.
|
||||||
<div>
|
</p>
|
||||||
<div id="piechart1" class="chart chart-50pc"></div>
|
<div class="charts">
|
||||||
<div id="piechart2" class="chart chart-50pc"></div>
|
<div id="piechart1" class="chart grid-left"></div>
|
||||||
</div>
|
<div id="piechart2" class="chart grid-right"></div>
|
||||||
<div id="barchart" class="chart"></div>
|
<div id="barchart" class="chart grid-full"></div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
<h2>Löschanfragen</h2>
|
<section>
|
||||||
<p>
|
<h2>Löschanfragen</h2>
|
||||||
Anfragen zur Löschung von Patientendaten, wenn kein Consent vorliegt.
|
<p>
|
||||||
</p>
|
Anfragen zur Löschung von Patientendaten, wenn kein Consent vorliegt.
|
||||||
<div>
|
</p>
|
||||||
<div id="piechartdel1" class="chart chart-50pc"></div>
|
<div class="charts">
|
||||||
<div id="piechartdel2" class="chart chart-50pc"></div>
|
<div id="piechartdel1" class="chart grid-left"></div>
|
||||||
</div>
|
<div id="piechartdel2" class="chart grid-right"></div>
|
||||||
<div id="barchartdel" class="chart"></div>
|
<div id="barchartdel" class="chart grid-full"></div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
</main>
|
</main>
|
||||||
<script th:src="@{/echarts.min.js}"></script>
|
<script th:src="@{/webjars/echarts/dist/echarts.min.js}"></script>
|
||||||
<script th:src="@{/scripts.js}"></script>
|
<script th:src="@{/scripts.js}"></script>
|
||||||
<script>
|
<script>
|
||||||
window.onload = () => {
|
window.onload = () => {
|
||||||
|
@ -0,0 +1,112 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.input
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import de.ukw.ccc.bwhc.dto.Consent
|
||||||
|
import de.ukw.ccc.bwhc.dto.MtbFile
|
||||||
|
import de.ukw.ccc.bwhc.dto.Patient
|
||||||
|
import dev.dnpm.etl.processor.services.RequestProcessor
|
||||||
|
import org.apache.kafka.clients.consumer.ConsumerRecord
|
||||||
|
import org.apache.kafka.common.header.internals.RecordHeader
|
||||||
|
import org.apache.kafka.common.header.internals.RecordHeaders
|
||||||
|
import org.apache.kafka.common.record.TimestampType
|
||||||
|
import org.junit.jupiter.api.BeforeEach
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith
|
||||||
|
import org.mockito.ArgumentMatchers.anyString
|
||||||
|
import org.mockito.Mock
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension
|
||||||
|
import org.mockito.kotlin.any
|
||||||
|
import org.mockito.kotlin.times
|
||||||
|
import org.mockito.kotlin.verify
|
||||||
|
import java.util.*
|
||||||
|
|
||||||
|
@ExtendWith(MockitoExtension::class)
|
||||||
|
class KafkaInputListenerTest {
|
||||||
|
|
||||||
|
private lateinit var requestProcessor: RequestProcessor
|
||||||
|
private lateinit var objectMapper: ObjectMapper
|
||||||
|
private lateinit var kafkaInputListener: KafkaInputListener
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
fun setup(
|
||||||
|
@Mock requestProcessor: RequestProcessor
|
||||||
|
) {
|
||||||
|
this.requestProcessor = requestProcessor
|
||||||
|
this.objectMapper = ObjectMapper()
|
||||||
|
|
||||||
|
this.kafkaInputListener = KafkaInputListener(requestProcessor, objectMapper)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldProcessMtbFileRequest() {
|
||||||
|
val mtbFile = MtbFile.builder()
|
||||||
|
.withPatient(Patient.builder().withId("DUMMY_12345678").build())
|
||||||
|
.withConsent(Consent.builder().withStatus(Consent.Status.ACTIVE).build())
|
||||||
|
.build()
|
||||||
|
|
||||||
|
kafkaInputListener.onMessage(ConsumerRecord("testtopic", 0, 0, "", this.objectMapper.writeValueAsString(mtbFile)))
|
||||||
|
|
||||||
|
verify(requestProcessor, times(1)).processMtbFile(any())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldProcessDeleteRequest() {
|
||||||
|
val mtbFile = MtbFile.builder()
|
||||||
|
.withPatient(Patient.builder().withId("DUMMY_12345678").build())
|
||||||
|
.withConsent(Consent.builder().withStatus(Consent.Status.REJECTED).build())
|
||||||
|
.build()
|
||||||
|
|
||||||
|
kafkaInputListener.onMessage(ConsumerRecord("testtopic", 0, 0, "", this.objectMapper.writeValueAsString(mtbFile)))
|
||||||
|
|
||||||
|
verify(requestProcessor, times(1)).processDeletion(anyString())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldProcessMtbFileRequestWithExistingRequestId() {
|
||||||
|
val mtbFile = MtbFile.builder()
|
||||||
|
.withPatient(Patient.builder().withId("DUMMY_12345678").build())
|
||||||
|
.withConsent(Consent.builder().withStatus(Consent.Status.ACTIVE).build())
|
||||||
|
.build()
|
||||||
|
|
||||||
|
val headers = RecordHeaders(listOf(RecordHeader("requestId", UUID.randomUUID().toString().toByteArray())))
|
||||||
|
kafkaInputListener.onMessage(
|
||||||
|
ConsumerRecord("testtopic", 0, 0, -1L, TimestampType.NO_TIMESTAMP_TYPE, -1, -1, "", this.objectMapper.writeValueAsString(mtbFile), headers, Optional.empty())
|
||||||
|
)
|
||||||
|
|
||||||
|
verify(requestProcessor, times(1)).processMtbFile(any(), anyString())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldProcessDeleteRequestWithExistingRequestId() {
|
||||||
|
val mtbFile = MtbFile.builder()
|
||||||
|
.withPatient(Patient.builder().withId("DUMMY_12345678").build())
|
||||||
|
.withConsent(Consent.builder().withStatus(Consent.Status.REJECTED).build())
|
||||||
|
.build()
|
||||||
|
|
||||||
|
val headers = RecordHeaders(listOf(RecordHeader("requestId", UUID.randomUUID().toString().toByteArray())))
|
||||||
|
kafkaInputListener.onMessage(
|
||||||
|
ConsumerRecord("testtopic", 0, 0, -1L, TimestampType.NO_TIMESTAMP_TYPE, -1, -1, "", this.objectMapper.writeValueAsString(mtbFile), headers, Optional.empty())
|
||||||
|
)
|
||||||
|
verify(requestProcessor, times(1)).processDeletion(anyString(), anyString())
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -17,7 +17,7 @@
|
|||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
package dev.dnpm.etl.processor.web
|
package dev.dnpm.etl.processor.input
|
||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
import de.ukw.ccc.bwhc.dto.*
|
import de.ukw.ccc.bwhc.dto.*
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -21,7 +21,7 @@ package dev.dnpm.etl.processor.output
|
|||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
import de.ukw.ccc.bwhc.dto.*
|
import de.ukw.ccc.bwhc.dto.*
|
||||||
import dev.dnpm.etl.processor.config.KafkaTargetProperties
|
import dev.dnpm.etl.processor.config.KafkaProperties
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
||||||
import org.assertj.core.api.Assertions.assertThat
|
import org.assertj.core.api.Assertions.assertThat
|
||||||
import org.junit.jupiter.api.BeforeEach
|
import org.junit.jupiter.api.BeforeEach
|
||||||
@ -35,6 +35,8 @@ import org.mockito.junit.jupiter.MockitoExtension
|
|||||||
import org.mockito.kotlin.*
|
import org.mockito.kotlin.*
|
||||||
import org.springframework.kafka.core.KafkaTemplate
|
import org.springframework.kafka.core.KafkaTemplate
|
||||||
import org.springframework.kafka.support.SendResult
|
import org.springframework.kafka.support.SendResult
|
||||||
|
import org.springframework.retry.policy.SimpleRetryPolicy
|
||||||
|
import org.springframework.retry.support.RetryTemplateBuilder
|
||||||
import java.util.concurrent.CompletableFuture.completedFuture
|
import java.util.concurrent.CompletableFuture.completedFuture
|
||||||
import java.util.concurrent.ExecutionException
|
import java.util.concurrent.ExecutionException
|
||||||
|
|
||||||
@ -51,11 +53,13 @@ class KafkaMtbFileSenderTest {
|
|||||||
fun setup(
|
fun setup(
|
||||||
@Mock kafkaTemplate: KafkaTemplate<String, String>
|
@Mock kafkaTemplate: KafkaTemplate<String, String>
|
||||||
) {
|
) {
|
||||||
val kafkaTargetProperties = KafkaTargetProperties("testtopic")
|
val kafkaProperties = KafkaProperties("testtopic")
|
||||||
|
val retryTemplate = RetryTemplateBuilder().customPolicy(SimpleRetryPolicy(1)).build()
|
||||||
|
|
||||||
this.objectMapper = ObjectMapper()
|
this.objectMapper = ObjectMapper()
|
||||||
this.kafkaTemplate = kafkaTemplate
|
this.kafkaTemplate = kafkaTemplate
|
||||||
|
|
||||||
this.kafkaMtbFileSender = KafkaMtbFileSender(kafkaTemplate, kafkaTargetProperties, objectMapper)
|
this.kafkaMtbFileSender = KafkaMtbFileSender(kafkaTemplate, kafkaProperties, retryTemplate, objectMapper)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ParameterizedTest
|
@ParameterizedTest
|
||||||
@ -97,9 +101,9 @@ class KafkaMtbFileSenderTest {
|
|||||||
val captor = argumentCaptor<String>()
|
val captor = argumentCaptor<String>()
|
||||||
verify(kafkaTemplate, times(1)).send(anyString(), captor.capture(), captor.capture())
|
verify(kafkaTemplate, times(1)).send(anyString(), captor.capture(), captor.capture())
|
||||||
assertThat(captor.firstValue).isNotNull
|
assertThat(captor.firstValue).isNotNull
|
||||||
assertThat(captor.firstValue).isEqualTo("{\"pid\": \"PID\", \"eid\": \"1\", \"requestId\": \"TestID\"}")
|
assertThat(captor.firstValue).isEqualTo("{\"pid\": \"PID\"}")
|
||||||
assertThat(captor.secondValue).isNotNull
|
assertThat(captor.secondValue).isNotNull
|
||||||
assertThat(captor.secondValue).isEqualTo(objectMapper.writeValueAsString(mtbFile(Consent.Status.ACTIVE)))
|
assertThat(captor.secondValue).isEqualTo(objectMapper.writeValueAsString(kafkaRecordData("TestID", Consent.Status.ACTIVE)))
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ -113,9 +117,61 @@ class KafkaMtbFileSenderTest {
|
|||||||
val captor = argumentCaptor<String>()
|
val captor = argumentCaptor<String>()
|
||||||
verify(kafkaTemplate, times(1)).send(anyString(), captor.capture(), captor.capture())
|
verify(kafkaTemplate, times(1)).send(anyString(), captor.capture(), captor.capture())
|
||||||
assertThat(captor.firstValue).isNotNull
|
assertThat(captor.firstValue).isNotNull
|
||||||
assertThat(captor.firstValue).isEqualTo("{\"pid\": \"PID\", \"requestId\": \"TestID\"}")
|
assertThat(captor.firstValue).isEqualTo("{\"pid\": \"PID\"}")
|
||||||
assertThat(captor.secondValue).isNotNull
|
assertThat(captor.secondValue).isNotNull
|
||||||
assertThat(captor.secondValue).isEqualTo(objectMapper.writeValueAsString(mtbFile(Consent.Status.REJECTED)))
|
assertThat(captor.secondValue).isEqualTo(objectMapper.writeValueAsString(kafkaRecordData("TestID", Consent.Status.REJECTED)))
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("requestWithResponseSource")
|
||||||
|
fun shouldRetryOnMtbFileKafkaSendError(testData: TestData) {
|
||||||
|
val kafkaProperties = KafkaProperties("testtopic")
|
||||||
|
val retryTemplate = RetryTemplateBuilder().customPolicy(SimpleRetryPolicy(3)).build()
|
||||||
|
this.kafkaMtbFileSender = KafkaMtbFileSender(this.kafkaTemplate, kafkaProperties, retryTemplate, this.objectMapper)
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
if (null != testData.exception) {
|
||||||
|
throw testData.exception
|
||||||
|
}
|
||||||
|
completedFuture(SendResult<String, String>(null, null))
|
||||||
|
}.whenever(kafkaTemplate).send(anyString(), anyString(), anyString())
|
||||||
|
|
||||||
|
kafkaMtbFileSender.send(MtbFileSender.MtbFileRequest("TestID", mtbFile(Consent.Status.ACTIVE)))
|
||||||
|
|
||||||
|
val expectedCount = when (testData.exception) {
|
||||||
|
// OK - No Retry
|
||||||
|
null -> times(1)
|
||||||
|
// Request failed - Retry max 3 times
|
||||||
|
else -> times(3)
|
||||||
|
}
|
||||||
|
|
||||||
|
verify(kafkaTemplate, expectedCount).send(anyString(), anyString(), anyString())
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("requestWithResponseSource")
|
||||||
|
fun shouldRetryOnDeleteKafkaSendError(testData: TestData) {
|
||||||
|
val kafkaProperties = KafkaProperties("testtopic")
|
||||||
|
val retryTemplate = RetryTemplateBuilder().customPolicy(SimpleRetryPolicy(3)).build()
|
||||||
|
this.kafkaMtbFileSender = KafkaMtbFileSender(this.kafkaTemplate, kafkaProperties, retryTemplate, this.objectMapper)
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
if (null != testData.exception) {
|
||||||
|
throw testData.exception
|
||||||
|
}
|
||||||
|
completedFuture(SendResult<String, String>(null, null))
|
||||||
|
}.whenever(kafkaTemplate).send(anyString(), anyString(), anyString())
|
||||||
|
|
||||||
|
kafkaMtbFileSender.send(MtbFileSender.DeleteRequest("TestID", "PID"))
|
||||||
|
|
||||||
|
val expectedCount = when (testData.exception) {
|
||||||
|
// OK - No Retry
|
||||||
|
null -> times(1)
|
||||||
|
// Request failed - Retry max 3 times
|
||||||
|
else -> times(3)
|
||||||
|
}
|
||||||
|
|
||||||
|
verify(kafkaTemplate, expectedCount).send(anyString(), anyString(), anyString())
|
||||||
}
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
@ -154,6 +210,10 @@ class KafkaMtbFileSenderTest {
|
|||||||
}.build()
|
}.build()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun kafkaRecordData(requestId: String, consentStatus: Consent.Status): KafkaMtbFileSender.Data {
|
||||||
|
return KafkaMtbFileSender.Data(requestId, mtbFile(consentStatus))
|
||||||
|
}
|
||||||
|
|
||||||
data class TestData(val requestStatus: RequestStatus, val exception: Throwable? = null)
|
data class TestData(val requestStatus: RequestStatus, val exception: Throwable? = null)
|
||||||
|
|
||||||
@JvmStatic
|
@JvmStatic
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* This file is part of ETL-Processor
|
* This file is part of ETL-Processor
|
||||||
*
|
*
|
||||||
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU Affero General Public License as published
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
@ -28,6 +28,9 @@ import org.junit.jupiter.params.ParameterizedTest
|
|||||||
import org.junit.jupiter.params.provider.MethodSource
|
import org.junit.jupiter.params.provider.MethodSource
|
||||||
import org.springframework.http.HttpMethod
|
import org.springframework.http.HttpMethod
|
||||||
import org.springframework.http.HttpStatus
|
import org.springframework.http.HttpStatus
|
||||||
|
import org.springframework.retry.policy.SimpleRetryPolicy
|
||||||
|
import org.springframework.retry.support.RetryTemplateBuilder
|
||||||
|
import org.springframework.test.web.client.ExpectedCount
|
||||||
import org.springframework.test.web.client.MockRestServiceServer
|
import org.springframework.test.web.client.MockRestServiceServer
|
||||||
import org.springframework.test.web.client.match.MockRestRequestMatchers.method
|
import org.springframework.test.web.client.match.MockRestRequestMatchers.method
|
||||||
import org.springframework.test.web.client.match.MockRestRequestMatchers.requestTo
|
import org.springframework.test.web.client.match.MockRestRequestMatchers.requestTo
|
||||||
@ -44,10 +47,11 @@ class RestMtbFileSenderTest {
|
|||||||
fun setup() {
|
fun setup() {
|
||||||
val restTemplate = RestTemplate()
|
val restTemplate = RestTemplate()
|
||||||
val restTargetProperties = RestTargetProperties("http://localhost:9000/mtbfile")
|
val restTargetProperties = RestTargetProperties("http://localhost:9000/mtbfile")
|
||||||
|
val retryTemplate = RetryTemplateBuilder().customPolicy(SimpleRetryPolicy(1)).build()
|
||||||
|
|
||||||
this.mockRestServiceServer = MockRestServiceServer.createServer(restTemplate)
|
this.mockRestServiceServer = MockRestServiceServer.createServer(restTemplate)
|
||||||
|
|
||||||
this.restMtbFileSender = RestMtbFileSender(restTemplate, restTargetProperties)
|
this.restMtbFileSender = RestMtbFileSender(restTemplate, restTargetProperties, retryTemplate)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ParameterizedTest
|
@ParameterizedTest
|
||||||
@ -80,6 +84,64 @@ class RestMtbFileSenderTest {
|
|||||||
assertThat(response.body).isEqualTo(requestWithResponse.response.body)
|
assertThat(response.body).isEqualTo(requestWithResponse.response.body)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("mtbFileRequestWithResponseSource")
|
||||||
|
fun shouldRetryOnMtbFileHttpRequestError(requestWithResponse: RequestWithResponse) {
|
||||||
|
val restTemplate = RestTemplate()
|
||||||
|
val restTargetProperties = RestTargetProperties("http://localhost:9000/mtbfile")
|
||||||
|
val retryTemplate = RetryTemplateBuilder().customPolicy(SimpleRetryPolicy(3)).build()
|
||||||
|
|
||||||
|
this.mockRestServiceServer = MockRestServiceServer.createServer(restTemplate)
|
||||||
|
this.restMtbFileSender = RestMtbFileSender(restTemplate, restTargetProperties, retryTemplate)
|
||||||
|
|
||||||
|
val expectedCount = when (requestWithResponse.httpStatus) {
|
||||||
|
// OK - No Retry
|
||||||
|
HttpStatus.OK, HttpStatus.CREATED -> ExpectedCount.max(1)
|
||||||
|
// Request failed - Retry max 3 times
|
||||||
|
else -> ExpectedCount.max(3)
|
||||||
|
}
|
||||||
|
|
||||||
|
this.mockRestServiceServer.expect(expectedCount) {
|
||||||
|
method(HttpMethod.POST)
|
||||||
|
requestTo("/mtbfile")
|
||||||
|
}.andRespond {
|
||||||
|
withStatus(requestWithResponse.httpStatus).body(requestWithResponse.body).createResponse(it)
|
||||||
|
}
|
||||||
|
|
||||||
|
val response = restMtbFileSender.send(MtbFileSender.MtbFileRequest("TestID", mtbFile))
|
||||||
|
assertThat(response.status).isEqualTo(requestWithResponse.response.status)
|
||||||
|
assertThat(response.body).isEqualTo(requestWithResponse.response.body)
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@MethodSource("deleteRequestWithResponseSource")
|
||||||
|
fun shouldRetryOnDeleteHttpRequestError(requestWithResponse: RequestWithResponse) {
|
||||||
|
val restTemplate = RestTemplate()
|
||||||
|
val restTargetProperties = RestTargetProperties("http://localhost:9000/mtbfile")
|
||||||
|
val retryTemplate = RetryTemplateBuilder().customPolicy(SimpleRetryPolicy(3)).build()
|
||||||
|
|
||||||
|
this.mockRestServiceServer = MockRestServiceServer.createServer(restTemplate)
|
||||||
|
this.restMtbFileSender = RestMtbFileSender(restTemplate, restTargetProperties, retryTemplate)
|
||||||
|
|
||||||
|
val expectedCount = when (requestWithResponse.httpStatus) {
|
||||||
|
// OK - No Retry
|
||||||
|
HttpStatus.OK, HttpStatus.CREATED -> ExpectedCount.max(1)
|
||||||
|
// Request failed - Retry max 3 times
|
||||||
|
else -> ExpectedCount.max(3)
|
||||||
|
}
|
||||||
|
|
||||||
|
this.mockRestServiceServer.expect(expectedCount) {
|
||||||
|
method(HttpMethod.DELETE)
|
||||||
|
requestTo("/mtbfile")
|
||||||
|
}.andRespond {
|
||||||
|
withStatus(requestWithResponse.httpStatus).body(requestWithResponse.body).createResponse(it)
|
||||||
|
}
|
||||||
|
|
||||||
|
val response = restMtbFileSender.send(MtbFileSender.DeleteRequest("TestID", "PID"))
|
||||||
|
assertThat(response.status).isEqualTo(requestWithResponse.response.status)
|
||||||
|
assertThat(response.body).isEqualTo(requestWithResponse.response.body)
|
||||||
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
data class RequestWithResponse(
|
data class RequestWithResponse(
|
||||||
val httpStatus: HttpStatus,
|
val httpStatus: HttpStatus,
|
||||||
@ -105,7 +167,7 @@ class RestMtbFileSenderTest {
|
|||||||
}
|
}
|
||||||
""".trimIndent()
|
""".trimIndent()
|
||||||
|
|
||||||
val mtbFile = MtbFile.builder()
|
val mtbFile: MtbFile = MtbFile.builder()
|
||||||
.withPatient(
|
.withPatient(
|
||||||
Patient.builder()
|
Patient.builder()
|
||||||
.withId("PID")
|
.withId("PID")
|
||||||
@ -129,7 +191,7 @@ class RestMtbFileSenderTest {
|
|||||||
)
|
)
|
||||||
.build()
|
.build()
|
||||||
|
|
||||||
private val errorResponseBody = "Sonstiger Fehler bei der Übertragung"
|
private const val ERROR_RESPONSE_BODY = "Sonstiger Fehler bei der Übertragung"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Synthetic http responses with related request status
|
* Synthetic http responses with related request status
|
||||||
@ -147,23 +209,23 @@ class RestMtbFileSenderTest {
|
|||||||
RequestWithResponse(
|
RequestWithResponse(
|
||||||
HttpStatus.BAD_REQUEST,
|
HttpStatus.BAD_REQUEST,
|
||||||
"??",
|
"??",
|
||||||
MtbFileSender.Response(RequestStatus.ERROR, errorResponseBody)
|
MtbFileSender.Response(RequestStatus.ERROR, ERROR_RESPONSE_BODY)
|
||||||
),
|
),
|
||||||
RequestWithResponse(
|
RequestWithResponse(
|
||||||
HttpStatus.UNPROCESSABLE_ENTITY,
|
HttpStatus.UNPROCESSABLE_ENTITY,
|
||||||
errorBody,
|
errorBody,
|
||||||
MtbFileSender.Response(RequestStatus.ERROR, errorResponseBody)
|
MtbFileSender.Response(RequestStatus.ERROR, ERROR_RESPONSE_BODY)
|
||||||
),
|
),
|
||||||
// Some more errors not mentioned in documentation
|
// Some more errors not mentioned in documentation
|
||||||
RequestWithResponse(
|
RequestWithResponse(
|
||||||
HttpStatus.NOT_FOUND,
|
HttpStatus.NOT_FOUND,
|
||||||
"what????",
|
"what????",
|
||||||
MtbFileSender.Response(RequestStatus.ERROR, errorResponseBody)
|
MtbFileSender.Response(RequestStatus.ERROR, ERROR_RESPONSE_BODY)
|
||||||
),
|
),
|
||||||
RequestWithResponse(
|
RequestWithResponse(
|
||||||
HttpStatus.INTERNAL_SERVER_ERROR,
|
HttpStatus.INTERNAL_SERVER_ERROR,
|
||||||
"what????",
|
"what????",
|
||||||
MtbFileSender.Response(RequestStatus.ERROR, errorResponseBody)
|
MtbFileSender.Response(RequestStatus.ERROR, ERROR_RESPONSE_BODY)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -180,12 +242,12 @@ class RestMtbFileSenderTest {
|
|||||||
RequestWithResponse(
|
RequestWithResponse(
|
||||||
HttpStatus.NOT_FOUND,
|
HttpStatus.NOT_FOUND,
|
||||||
"what????",
|
"what????",
|
||||||
MtbFileSender.Response(RequestStatus.ERROR, errorResponseBody)
|
MtbFileSender.Response(RequestStatus.ERROR, ERROR_RESPONSE_BODY)
|
||||||
),
|
),
|
||||||
RequestWithResponse(
|
RequestWithResponse(
|
||||||
HttpStatus.INTERNAL_SERVER_ERROR,
|
HttpStatus.INTERNAL_SERVER_ERROR,
|
||||||
"what????",
|
"what????",
|
||||||
MtbFileSender.Response(RequestStatus.ERROR, errorResponseBody)
|
MtbFileSender.Response(RequestStatus.ERROR, ERROR_RESPONSE_BODY)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,198 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.pseudonym
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import de.ukw.ccc.bwhc.dto.*
|
||||||
|
import org.assertj.core.api.Assertions.assertThat
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
import org.junit.jupiter.api.assertThrows
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith
|
||||||
|
import org.mockito.ArgumentMatchers
|
||||||
|
import org.mockito.Mock
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension
|
||||||
|
import org.mockito.kotlin.doAnswer
|
||||||
|
import org.mockito.kotlin.whenever
|
||||||
|
import org.springframework.core.io.ClassPathResource
|
||||||
|
|
||||||
|
const val FAKE_MTB_FILE_PATH = "fake_MTBFile.json"
|
||||||
|
const val CLEAN_PATIENT_ID = "5dad2f0b-49c6-47d8-a952-7b9e9e0f7549"
|
||||||
|
|
||||||
|
@ExtendWith(MockitoExtension::class)
|
||||||
|
class ExtensionsTest {
|
||||||
|
|
||||||
|
private fun fakeMtbFile(): MtbFile {
|
||||||
|
val mtbFile = ClassPathResource(FAKE_MTB_FILE_PATH).inputStream
|
||||||
|
return ObjectMapper().readValue(mtbFile, MtbFile::class.java)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun MtbFile.serialized(): String {
|
||||||
|
return ObjectMapper().writeValueAsString(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldNotContainCleanPatientId(@Mock pseudonymizeService: PseudonymizeService) {
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
"PSEUDO-ID"
|
||||||
|
}.whenever(pseudonymizeService).patientPseudonym(ArgumentMatchers.anyString())
|
||||||
|
|
||||||
|
val mtbFile = fakeMtbFile()
|
||||||
|
|
||||||
|
mtbFile.pseudonymizeWith(pseudonymizeService)
|
||||||
|
|
||||||
|
assertThat(mtbFile.patient.id).isEqualTo("PSEUDO-ID")
|
||||||
|
assertThat(mtbFile.serialized()).doesNotContain(CLEAN_PATIENT_ID)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldNotContainAnyUuidAfterRehashingOfIds(@Mock pseudonymizeService: PseudonymizeService) {
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
"PSEUDO-ID"
|
||||||
|
}.whenever(pseudonymizeService).patientPseudonym(ArgumentMatchers.anyString())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
"TESTDOMAIN"
|
||||||
|
}.whenever(pseudonymizeService).prefix()
|
||||||
|
|
||||||
|
val mtbFile = fakeMtbFile()
|
||||||
|
|
||||||
|
mtbFile.pseudonymizeWith(pseudonymizeService)
|
||||||
|
mtbFile.anonymizeContentWith(pseudonymizeService)
|
||||||
|
|
||||||
|
val pattern = "\"[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}\"".toRegex().toPattern()
|
||||||
|
val matcher = pattern.matcher(mtbFile.serialized())
|
||||||
|
|
||||||
|
assertThrows<IllegalStateException> {
|
||||||
|
matcher.find()
|
||||||
|
matcher.group()
|
||||||
|
}.also {
|
||||||
|
assertThat(it.message).isEqualTo("No match found")
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldRehashIdsWithPrefix(@Mock pseudonymizeService: PseudonymizeService) {
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
"PSEUDO-ID"
|
||||||
|
}.whenever(pseudonymizeService).patientPseudonym(ArgumentMatchers.anyString())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
"TESTDOMAIN"
|
||||||
|
}.whenever(pseudonymizeService).prefix()
|
||||||
|
|
||||||
|
val mtbFile = MtbFile.builder()
|
||||||
|
.withPatient(
|
||||||
|
Patient.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withBirthDate("2000-08-08")
|
||||||
|
.withGender(Patient.Gender.MALE)
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withConsent(
|
||||||
|
Consent.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withStatus(Consent.Status.ACTIVE)
|
||||||
|
.withPatient("123")
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withEpisode(
|
||||||
|
Episode.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withPatient("1")
|
||||||
|
.withPeriod(PeriodStart("2023-08-08"))
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.build()
|
||||||
|
|
||||||
|
mtbFile.pseudonymizeWith(pseudonymizeService)
|
||||||
|
mtbFile.anonymizeContentWith(pseudonymizeService)
|
||||||
|
|
||||||
|
|
||||||
|
assertThat(mtbFile.episode.id)
|
||||||
|
// TESTDOMAIN<sha256(TESTDOMAIN-1)[0-41]>
|
||||||
|
.isEqualTo("TESTDOMAIN44e20a53bbbf9f3ae39626d05df7014dcd77d6098")
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldNotThrowExceptionOnNullValues(@Mock pseudonymizeService: PseudonymizeService) {
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
"PSEUDO-ID"
|
||||||
|
}.whenever(pseudonymizeService).patientPseudonym(ArgumentMatchers.anyString())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
"TESTDOMAIN"
|
||||||
|
}.whenever(pseudonymizeService).prefix()
|
||||||
|
|
||||||
|
val mtbFile = MtbFile.builder()
|
||||||
|
.withPatient(
|
||||||
|
Patient.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withBirthDate("2000-08-08")
|
||||||
|
.withGender(Patient.Gender.MALE)
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withConsent(
|
||||||
|
Consent.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withStatus(Consent.Status.ACTIVE)
|
||||||
|
.withPatient("123")
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withEpisode(
|
||||||
|
Episode.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withPatient("1")
|
||||||
|
.withPeriod(PeriodStart("2023-08-08"))
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withClaims(null)
|
||||||
|
.withDiagnoses(null)
|
||||||
|
.withCarePlans(null)
|
||||||
|
.withClaimResponses(null)
|
||||||
|
.withEcogStatus(null)
|
||||||
|
.withFamilyMemberDiagnoses(null)
|
||||||
|
.withGeneticCounsellingRequests(null)
|
||||||
|
.withHistologyReevaluationRequests(null)
|
||||||
|
.withHistologyReports(null)
|
||||||
|
.withLastGuidelineTherapies(null)
|
||||||
|
.withMolecularPathologyFindings(null)
|
||||||
|
.withMolecularTherapies(null)
|
||||||
|
.withNgsReports(null)
|
||||||
|
.withPreviousGuidelineTherapies(null)
|
||||||
|
.withRebiopsyRequests(null)
|
||||||
|
.withRecommendations(null)
|
||||||
|
.withResponses(null)
|
||||||
|
.withStudyInclusionRequests(null)
|
||||||
|
.withSpecimens(null)
|
||||||
|
.build()
|
||||||
|
|
||||||
|
mtbFile.pseudonymizeWith(pseudonymizeService)
|
||||||
|
mtbFile.anonymizeContentWith(pseudonymizeService)
|
||||||
|
|
||||||
|
|
||||||
|
assertThat(mtbFile.episode.id).isNotNull()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -70,6 +70,13 @@ class PseudonymizeServiceTest {
|
|||||||
assertThat(mtbFile.patient.id).isEqualTo("123")
|
assertThat(mtbFile.patient.id).isEqualTo("123")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun sanitizeFileName(@Mock generator: GpasPseudonymGenerator) {
|
||||||
|
val result= GpasPseudonymGenerator.sanitizeValue("l://a\\bs;1*2?3>")
|
||||||
|
|
||||||
|
assertThat(result).isEqualTo("l___a_bs_1_2_3_")
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun shouldUsePseudonymPrefixForBuiltin(@Mock generator: AnonymizingGenerator) {
|
fun shouldUsePseudonymPrefixForBuiltin(@Mock generator: AnonymizingGenerator) {
|
||||||
doAnswer {
|
doAnswer {
|
||||||
|
@ -41,19 +41,25 @@ class ReportServiceTest {
|
|||||||
{
|
{
|
||||||
"patient": "4711",
|
"patient": "4711",
|
||||||
"issues": [
|
"issues": [
|
||||||
|
{ "severity": "info", "message": "Info Message" },
|
||||||
{ "severity": "warning", "message": "Warning Message" },
|
{ "severity": "warning", "message": "Warning Message" },
|
||||||
{ "severity": "error", "message": "Error Message" }
|
{ "severity": "error", "message": "Error Message" },
|
||||||
|
{ "severity": "fatal", "message": "Fatal Message" }
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
""".trimIndent()
|
""".trimIndent()
|
||||||
|
|
||||||
val actual = this.reportService.deserialize(json)
|
val actual = this.reportService.deserialize(json)
|
||||||
|
|
||||||
assertThat(actual).hasSize(2)
|
assertThat(actual).hasSize(4)
|
||||||
assertThat(actual[0].severity).isEqualTo(ReportService.Severity.WARNING)
|
assertThat(actual[0].severity).isEqualTo(ReportService.Severity.FATAL)
|
||||||
assertThat(actual[0].message).isEqualTo("Warning Message")
|
assertThat(actual[0].message).isEqualTo("Fatal Message")
|
||||||
assertThat(actual[1].severity).isEqualTo(ReportService.Severity.ERROR)
|
assertThat(actual[1].severity).isEqualTo(ReportService.Severity.ERROR)
|
||||||
assertThat(actual[1].message).isEqualTo("Error Message")
|
assertThat(actual[1].message).isEqualTo("Error Message")
|
||||||
|
assertThat(actual[2].severity).isEqualTo(ReportService.Severity.WARNING)
|
||||||
|
assertThat(actual[2].message).isEqualTo("Warning Message")
|
||||||
|
assertThat(actual[3].severity).isEqualTo(ReportService.Severity.INFO)
|
||||||
|
assertThat(actual[3].message).isEqualTo("Info Message")
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -21,6 +21,7 @@ package dev.dnpm.etl.processor.services
|
|||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
import de.ukw.ccc.bwhc.dto.*
|
import de.ukw.ccc.bwhc.dto.*
|
||||||
|
import dev.dnpm.etl.processor.config.AppConfigProperties
|
||||||
import dev.dnpm.etl.processor.monitoring.Request
|
import dev.dnpm.etl.processor.monitoring.Request
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestType
|
import dev.dnpm.etl.processor.monitoring.RequestType
|
||||||
@ -37,6 +38,7 @@ import org.mockito.Mockito.*
|
|||||||
import org.mockito.junit.jupiter.MockitoExtension
|
import org.mockito.junit.jupiter.MockitoExtension
|
||||||
import org.mockito.kotlin.any
|
import org.mockito.kotlin.any
|
||||||
import org.mockito.kotlin.argumentCaptor
|
import org.mockito.kotlin.argumentCaptor
|
||||||
|
import org.mockito.kotlin.whenever
|
||||||
import org.springframework.context.ApplicationEventPublisher
|
import org.springframework.context.ApplicationEventPublisher
|
||||||
import java.time.Instant
|
import java.time.Instant
|
||||||
import java.util.*
|
import java.util.*
|
||||||
@ -46,32 +48,39 @@ import java.util.*
|
|||||||
class RequestProcessorTest {
|
class RequestProcessorTest {
|
||||||
|
|
||||||
private lateinit var pseudonymizeService: PseudonymizeService
|
private lateinit var pseudonymizeService: PseudonymizeService
|
||||||
|
private lateinit var transformationService: TransformationService
|
||||||
private lateinit var sender: MtbFileSender
|
private lateinit var sender: MtbFileSender
|
||||||
private lateinit var requestService: RequestService
|
private lateinit var requestService: RequestService
|
||||||
private lateinit var applicationEventPublisher: ApplicationEventPublisher
|
private lateinit var applicationEventPublisher: ApplicationEventPublisher
|
||||||
|
private lateinit var appConfigProperties: AppConfigProperties
|
||||||
|
|
||||||
private lateinit var requestProcessor: RequestProcessor
|
private lateinit var requestProcessor: RequestProcessor
|
||||||
|
|
||||||
@BeforeEach
|
@BeforeEach
|
||||||
fun setup(
|
fun setup(
|
||||||
@Mock pseudonymizeService: PseudonymizeService,
|
@Mock pseudonymizeService: PseudonymizeService,
|
||||||
|
@Mock transformationService: TransformationService,
|
||||||
@Mock sender: RestMtbFileSender,
|
@Mock sender: RestMtbFileSender,
|
||||||
@Mock requestService: RequestService,
|
@Mock requestService: RequestService,
|
||||||
@Mock applicationEventPublisher: ApplicationEventPublisher
|
@Mock applicationEventPublisher: ApplicationEventPublisher
|
||||||
) {
|
) {
|
||||||
this.pseudonymizeService = pseudonymizeService
|
this.pseudonymizeService = pseudonymizeService
|
||||||
|
this.transformationService = transformationService
|
||||||
this.sender = sender
|
this.sender = sender
|
||||||
this.requestService = requestService
|
this.requestService = requestService
|
||||||
this.applicationEventPublisher = applicationEventPublisher
|
this.applicationEventPublisher = applicationEventPublisher
|
||||||
|
this.appConfigProperties = AppConfigProperties(null)
|
||||||
|
|
||||||
val objectMapper = ObjectMapper()
|
val objectMapper = ObjectMapper()
|
||||||
|
|
||||||
requestProcessor = RequestProcessor(
|
requestProcessor = RequestProcessor(
|
||||||
pseudonymizeService,
|
pseudonymizeService,
|
||||||
|
transformationService,
|
||||||
sender,
|
sender,
|
||||||
requestService,
|
requestService,
|
||||||
objectMapper,
|
objectMapper,
|
||||||
applicationEventPublisher
|
applicationEventPublisher,
|
||||||
|
appConfigProperties
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -83,7 +92,7 @@ class RequestProcessorTest {
|
|||||||
uuid = UUID.randomUUID().toString(),
|
uuid = UUID.randomUUID().toString(),
|
||||||
patientId = "TEST_12345678901",
|
patientId = "TEST_12345678901",
|
||||||
pid = "P1",
|
pid = "P1",
|
||||||
fingerprint = "xrysxpozhbs2lnrjgf3yq4fzj33kxr7xr5c2cbuskmelfdmckl3a",
|
fingerprint = "zdlzv5s5ydmd4ktw2v5piohegc4jcyrm6j66bq6tv2uxuerndmga",
|
||||||
type = RequestType.MTB_FILE,
|
type = RequestType.MTB_FILE,
|
||||||
status = RequestStatus.SUCCESS,
|
status = RequestStatus.SUCCESS,
|
||||||
processedAt = Instant.parse("2023-08-08T02:00:00Z")
|
processedAt = Instant.parse("2023-08-08T02:00:00Z")
|
||||||
@ -98,6 +107,10 @@ class RequestProcessorTest {
|
|||||||
it.arguments[0] as String
|
it.arguments[0] as String
|
||||||
}.`when`(pseudonymizeService).patientPseudonym(any())
|
}.`when`(pseudonymizeService).patientPseudonym(any())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
}.whenever(transformationService).transform(any())
|
||||||
|
|
||||||
val mtbFile = MtbFile.builder()
|
val mtbFile = MtbFile.builder()
|
||||||
.withPatient(
|
.withPatient(
|
||||||
Patient.builder()
|
Patient.builder()
|
||||||
@ -138,7 +151,7 @@ class RequestProcessorTest {
|
|||||||
uuid = UUID.randomUUID().toString(),
|
uuid = UUID.randomUUID().toString(),
|
||||||
patientId = "TEST_12345678901",
|
patientId = "TEST_12345678901",
|
||||||
pid = "P1",
|
pid = "P1",
|
||||||
fingerprint = "xrysxpozhbs2lnrjgf3yq4fzj33kxr7xr5c2cbuskmelfdmckl3a",
|
fingerprint = "zdlzv5s5ydmd4ktw2v5piohegc4jcyrm6j66bq6tv2uxuerndmga",
|
||||||
type = RequestType.MTB_FILE,
|
type = RequestType.MTB_FILE,
|
||||||
status = RequestStatus.SUCCESS,
|
status = RequestStatus.SUCCESS,
|
||||||
processedAt = Instant.parse("2023-08-08T02:00:00Z")
|
processedAt = Instant.parse("2023-08-08T02:00:00Z")
|
||||||
@ -153,6 +166,10 @@ class RequestProcessorTest {
|
|||||||
it.arguments[0] as String
|
it.arguments[0] as String
|
||||||
}.`when`(pseudonymizeService).patientPseudonym(any())
|
}.`when`(pseudonymizeService).patientPseudonym(any())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
}.whenever(transformationService).transform(any())
|
||||||
|
|
||||||
val mtbFile = MtbFile.builder()
|
val mtbFile = MtbFile.builder()
|
||||||
.withPatient(
|
.withPatient(
|
||||||
Patient.builder()
|
Patient.builder()
|
||||||
@ -212,6 +229,10 @@ class RequestProcessorTest {
|
|||||||
it.arguments[0] as String
|
it.arguments[0] as String
|
||||||
}.`when`(pseudonymizeService).patientPseudonym(any())
|
}.`when`(pseudonymizeService).patientPseudonym(any())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
}.whenever(transformationService).transform(any())
|
||||||
|
|
||||||
val mtbFile = MtbFile.builder()
|
val mtbFile = MtbFile.builder()
|
||||||
.withPatient(
|
.withPatient(
|
||||||
Patient.builder()
|
Patient.builder()
|
||||||
@ -271,6 +292,10 @@ class RequestProcessorTest {
|
|||||||
it.arguments[0] as String
|
it.arguments[0] as String
|
||||||
}.`when`(pseudonymizeService).patientPseudonym(any())
|
}.`when`(pseudonymizeService).patientPseudonym(any())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
}.whenever(transformationService).transform(any())
|
||||||
|
|
||||||
val mtbFile = MtbFile.builder()
|
val mtbFile = MtbFile.builder()
|
||||||
.withPatient(
|
.withPatient(
|
||||||
Patient.builder()
|
Patient.builder()
|
||||||
@ -369,4 +394,52 @@ class RequestProcessorTest {
|
|||||||
assertThat(requestCaptor.firstValue.status).isEqualTo(RequestStatus.ERROR)
|
assertThat(requestCaptor.firstValue.status).isEqualTo(RequestStatus.ERROR)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun testShouldNotDetectMtbFileDuplicationIfDuplicationNotConfigured() {
|
||||||
|
this.appConfigProperties.duplicationDetection = false
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0] as String
|
||||||
|
}.`when`(pseudonymizeService).patientPseudonym(any())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
it.arguments[0]
|
||||||
|
}.whenever(transformationService).transform(any())
|
||||||
|
|
||||||
|
doAnswer {
|
||||||
|
MtbFileSender.Response(status = RequestStatus.SUCCESS)
|
||||||
|
}.`when`(sender).send(any<MtbFileSender.MtbFileRequest>())
|
||||||
|
|
||||||
|
val mtbFile = MtbFile.builder()
|
||||||
|
.withPatient(
|
||||||
|
Patient.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withBirthDate("2000-08-08")
|
||||||
|
.withGender(Patient.Gender.MALE)
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withConsent(
|
||||||
|
Consent.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withStatus(Consent.Status.ACTIVE)
|
||||||
|
.withPatient("123")
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.withEpisode(
|
||||||
|
Episode.builder()
|
||||||
|
.withId("1")
|
||||||
|
.withPatient("1")
|
||||||
|
.withPeriod(PeriodStart("2023-08-08"))
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
.build()
|
||||||
|
|
||||||
|
this.requestProcessor.processMtbFile(mtbFile)
|
||||||
|
|
||||||
|
val eventCaptor = argumentCaptor<ResponseEvent>()
|
||||||
|
verify(applicationEventPublisher, times(1)).publishEvent(eventCaptor.capture())
|
||||||
|
assertThat(eventCaptor.firstValue).isNotNull
|
||||||
|
assertThat(eventCaptor.firstValue.status).isEqualTo(RequestStatus.SUCCESS)
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
@ -19,8 +19,6 @@
|
|||||||
|
|
||||||
package dev.dnpm.etl.processor.services
|
package dev.dnpm.etl.processor.services
|
||||||
|
|
||||||
import com.fasterxml.jackson.databind.ObjectMapper
|
|
||||||
import com.fasterxml.jackson.module.kotlin.KotlinModule
|
|
||||||
import dev.dnpm.etl.processor.monitoring.Request
|
import dev.dnpm.etl.processor.monitoring.Request
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
import dev.dnpm.etl.processor.monitoring.RequestRepository
|
||||||
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
import dev.dnpm.etl.processor.monitoring.RequestStatus
|
||||||
@ -62,12 +60,10 @@ class ResponseProcessorTest {
|
|||||||
@Mock requestRepository: RequestRepository,
|
@Mock requestRepository: RequestRepository,
|
||||||
@Mock statisticsUpdateProducer: Sinks.Many<Any>
|
@Mock statisticsUpdateProducer: Sinks.Many<Any>
|
||||||
) {
|
) {
|
||||||
val objectMapper = ObjectMapper().registerModule(KotlinModule.Builder().build())
|
|
||||||
|
|
||||||
this.requestRepository = requestRepository
|
this.requestRepository = requestRepository
|
||||||
this.statisticsUpdateProducer = statisticsUpdateProducer
|
this.statisticsUpdateProducer = statisticsUpdateProducer
|
||||||
|
|
||||||
this.responseProcessor = ResponseProcessor(requestRepository, statisticsUpdateProducer, objectMapper)
|
this.responseProcessor = ResponseProcessor(requestRepository, statisticsUpdateProducer)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ -0,0 +1,154 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2024 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.services
|
||||||
|
|
||||||
|
import org.assertj.core.api.Assertions.assertThat
|
||||||
|
import org.junit.jupiter.api.BeforeEach
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith
|
||||||
|
import org.mockito.ArgumentCaptor
|
||||||
|
import org.mockito.ArgumentMatchers.anyLong
|
||||||
|
import org.mockito.ArgumentMatchers.anyString
|
||||||
|
import org.mockito.Mock
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension
|
||||||
|
import org.mockito.kotlin.*
|
||||||
|
import org.springframework.security.crypto.password.PasswordEncoder
|
||||||
|
import org.springframework.security.provisioning.InMemoryUserDetailsManager
|
||||||
|
import java.util.*
|
||||||
|
import java.util.function.Consumer
|
||||||
|
|
||||||
|
@ExtendWith(MockitoExtension::class)
|
||||||
|
class TokenServiceTest {
|
||||||
|
|
||||||
|
private lateinit var userDetailsManager: InMemoryUserDetailsManager
|
||||||
|
private lateinit var passwordEncoder: PasswordEncoder
|
||||||
|
private lateinit var tokenRepository: TokenRepository
|
||||||
|
|
||||||
|
private lateinit var tokenService: TokenService
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
fun setup(
|
||||||
|
@Mock userDetailsManager: InMemoryUserDetailsManager,
|
||||||
|
@Mock passwordEncoder: PasswordEncoder,
|
||||||
|
@Mock tokenRepository: TokenRepository
|
||||||
|
) {
|
||||||
|
this.userDetailsManager = userDetailsManager
|
||||||
|
this.passwordEncoder = passwordEncoder
|
||||||
|
this.tokenRepository = tokenRepository
|
||||||
|
|
||||||
|
this.tokenService = TokenService(userDetailsManager, passwordEncoder, tokenRepository)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldEncodePasswordForNewToken() {
|
||||||
|
doAnswer { "{test}verysecret" }.whenever(passwordEncoder).encode(anyString())
|
||||||
|
|
||||||
|
val actual = this.tokenService.addToken("Test Token")
|
||||||
|
|
||||||
|
assertThat(actual).satisfies(
|
||||||
|
Consumer { assertThat(it.isSuccess).isTrue() },
|
||||||
|
Consumer { assertThat(it.getOrNull()).matches("testtoken:[A-Za-z0-9]{48}$") }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldContainAlphanumTokenUserPart() {
|
||||||
|
doAnswer { "{test}verysecret" }.whenever(passwordEncoder).encode(anyString())
|
||||||
|
|
||||||
|
val actual = this.tokenService.addToken("Test Token")
|
||||||
|
|
||||||
|
assertThat(actual).satisfies(
|
||||||
|
Consumer { assertThat(it.isSuccess).isTrue() },
|
||||||
|
Consumer { assertThat(it.getOrDefault("")).startsWith("testtoken:") }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldNotAllowSameTokenUserPartTwice() {
|
||||||
|
doReturn(true).whenever(userDetailsManager).userExists(anyString())
|
||||||
|
|
||||||
|
val actual = this.tokenService.addToken("Test Token")
|
||||||
|
|
||||||
|
assertThat(actual).satisfies(Consumer { assertThat(it.isFailure).isTrue() })
|
||||||
|
verify(tokenRepository, never()).save(any())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldSaveNewToken() {
|
||||||
|
doAnswer { "{test}verysecret" }.whenever(passwordEncoder).encode(anyString())
|
||||||
|
|
||||||
|
val actual = this.tokenService.addToken("Test Token")
|
||||||
|
|
||||||
|
val captor = ArgumentCaptor.forClass(Token::class.java)
|
||||||
|
verify(tokenRepository, times(1)).save(captor.capture())
|
||||||
|
|
||||||
|
assertThat(actual).satisfies(Consumer { assertThat(it.isSuccess).isTrue() })
|
||||||
|
assertThat(captor.value).satisfies(
|
||||||
|
Consumer { assertThat(it.name).isEqualTo("Test Token") },
|
||||||
|
Consumer { assertThat(it.username).isEqualTo("testtoken") },
|
||||||
|
Consumer { assertThat(it.password).isEqualTo("{test}verysecret") }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldDeleteExistingToken() {
|
||||||
|
doAnswer {
|
||||||
|
val id = it.arguments[0] as Long
|
||||||
|
Optional.of(Token(id, "Test Token", "testtoken", "{test}hsdajfgadskjhfgsdkfjg"))
|
||||||
|
}.whenever(tokenRepository).findById(anyLong())
|
||||||
|
|
||||||
|
this.tokenService.deleteToken(42)
|
||||||
|
|
||||||
|
val stringCaptor = ArgumentCaptor.forClass(String::class.java)
|
||||||
|
verify(userDetailsManager, times(1)).deleteUser(stringCaptor.capture())
|
||||||
|
assertThat(stringCaptor.value).isEqualTo("testtoken")
|
||||||
|
|
||||||
|
val tokenCaptor = ArgumentCaptor.forClass(Token::class.java)
|
||||||
|
verify(tokenRepository, times(1)).delete(tokenCaptor.capture())
|
||||||
|
assertThat(tokenCaptor.value.id).isEqualTo(42)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldReturnAllTokensFromRepository() {
|
||||||
|
val expected = listOf(
|
||||||
|
Token(1, "Test Token 1", "testtoken1", "{test}hsdajfgadskjhfgsdkfjg"),
|
||||||
|
Token(2, "Test Token 2", "testtoken2", "{test}asdasdasdasdasdasdasd")
|
||||||
|
)
|
||||||
|
|
||||||
|
doReturn(expected).whenever(tokenRepository).findAll()
|
||||||
|
|
||||||
|
assertThat(tokenService.findAll()).isEqualTo(expected)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldAddAllTokensFromRepositoryToUserDataManager() {
|
||||||
|
val expected = listOf(
|
||||||
|
Token(1, "Test Token 1", "testtoken1", "{test}hsdajfgadskjhfgsdkfjg"),
|
||||||
|
Token(2, "Test Token 2", "testtoken2", "{test}asdasdasdasdasdasdasd")
|
||||||
|
)
|
||||||
|
|
||||||
|
doReturn(expected).whenever(tokenRepository).findAll()
|
||||||
|
|
||||||
|
tokenService.setup()
|
||||||
|
|
||||||
|
verify(userDetailsManager, times(expected.size)).createUser(any())
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,95 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ETL-Processor
|
||||||
|
*
|
||||||
|
* Copyright (c) 2023 Comprehensive Cancer Center Mainfranken, Datenintegrationszentrum Philipps-Universität Marburg and Contributors
|
||||||
|
*
|
||||||
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU Affero General Public License as published
|
||||||
|
* by the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU Affero General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Affero General Public License
|
||||||
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dev.dnpm.etl.processor.services
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper
|
||||||
|
import de.ukw.ccc.bwhc.dto.Consent
|
||||||
|
import de.ukw.ccc.bwhc.dto.Diagnosis
|
||||||
|
import de.ukw.ccc.bwhc.dto.Icd10
|
||||||
|
import de.ukw.ccc.bwhc.dto.MtbFile
|
||||||
|
import org.assertj.core.api.Assertions.assertThat
|
||||||
|
import org.junit.jupiter.api.BeforeEach
|
||||||
|
import org.junit.jupiter.api.Test
|
||||||
|
|
||||||
|
class TransformationServiceTest {
|
||||||
|
|
||||||
|
private lateinit var service: TransformationService
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
fun setup() {
|
||||||
|
this.service = TransformationService(
|
||||||
|
ObjectMapper(), listOf(
|
||||||
|
Transformation.of("consent.status") from Consent.Status.ACTIVE to Consent.Status.REJECTED,
|
||||||
|
Transformation.of("diagnoses[*].icd10.version") from "2013" to "2014",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldTransformMtbFile() {
|
||||||
|
val mtbFile = MtbFile.builder().withDiagnoses(
|
||||||
|
listOf(
|
||||||
|
Diagnosis.builder().withId("1234").withIcd10(Icd10("F79.9").also {
|
||||||
|
it.version = "2013"
|
||||||
|
}).build()
|
||||||
|
)
|
||||||
|
).build()
|
||||||
|
|
||||||
|
val actual = this.service.transform(mtbFile)
|
||||||
|
|
||||||
|
assertThat(actual).isNotNull
|
||||||
|
assertThat(actual.diagnoses[0].icd10.version).isEqualTo("2014")
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldOnlyTransformGivenValues() {
|
||||||
|
val mtbFile = MtbFile.builder().withDiagnoses(
|
||||||
|
listOf(
|
||||||
|
Diagnosis.builder().withId("1234").withIcd10(Icd10("F79.9").also {
|
||||||
|
it.version = "2013"
|
||||||
|
}).build(),
|
||||||
|
Diagnosis.builder().withId("5678").withIcd10(Icd10("F79.8").also {
|
||||||
|
it.version = "2019"
|
||||||
|
}).build()
|
||||||
|
)
|
||||||
|
).build()
|
||||||
|
|
||||||
|
val actual = this.service.transform(mtbFile)
|
||||||
|
|
||||||
|
assertThat(actual).isNotNull
|
||||||
|
assertThat(actual.diagnoses[0].icd10.code).isEqualTo("F79.9")
|
||||||
|
assertThat(actual.diagnoses[0].icd10.version).isEqualTo("2014")
|
||||||
|
assertThat(actual.diagnoses[1].icd10.code).isEqualTo("F79.8")
|
||||||
|
assertThat(actual.diagnoses[1].icd10.version).isEqualTo("2019")
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldTransformMtbFileWithConsentEnum() {
|
||||||
|
val mtbFile = MtbFile.builder().withConsent(
|
||||||
|
Consent("123", "456", Consent.Status.ACTIVE)
|
||||||
|
).build()
|
||||||
|
|
||||||
|
val actual = this.service.transform(mtbFile)
|
||||||
|
|
||||||
|
assertThat(actual.consent).isNotNull
|
||||||
|
assertThat(actual.consent.status).isEqualTo(Consent.Status.REJECTED)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -45,8 +45,8 @@ class KafkaResponseProcessorTest {
|
|||||||
|
|
||||||
private lateinit var kafkaResponseProcessor: KafkaResponseProcessor
|
private lateinit var kafkaResponseProcessor: KafkaResponseProcessor
|
||||||
|
|
||||||
private fun createkafkaRecord(
|
private fun createKafkaRecord(
|
||||||
requestId: String? = null,
|
requestId: String,
|
||||||
statusCode: Int = 200,
|
statusCode: Int = 200,
|
||||||
statusBody: Map<String, Any>? = mapOf()
|
statusBody: Map<String, Any>? = mapOf()
|
||||||
): ConsumerRecord<String, String> {
|
): ConsumerRecord<String, String> {
|
||||||
@ -54,15 +54,11 @@ class KafkaResponseProcessorTest {
|
|||||||
"test-topic",
|
"test-topic",
|
||||||
0,
|
0,
|
||||||
0,
|
0,
|
||||||
if (requestId == null) {
|
null,
|
||||||
null
|
|
||||||
} else {
|
|
||||||
this.objectMapper.writeValueAsString(KafkaResponseProcessor.ResponseKey(requestId))
|
|
||||||
},
|
|
||||||
if (statusBody == null) {
|
if (statusBody == null) {
|
||||||
""
|
""
|
||||||
} else {
|
} else {
|
||||||
this.objectMapper.writeValueAsString(KafkaResponseProcessor.ResponseBody(statusCode, statusBody))
|
this.objectMapper.writeValueAsString(KafkaResponseProcessor.ResponseBody(requestId, statusCode, statusBody))
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -78,23 +74,57 @@ class KafkaResponseProcessorTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun shouldNotProcessRecordsWithoutValidKey() {
|
fun shouldNotProcessRecordsWithoutRequestIdInBody() {
|
||||||
this.kafkaResponseProcessor.onMessage(createkafkaRecord(null, 200))
|
val record = ConsumerRecord<String, String>(
|
||||||
|
"test-topic",
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
null,
|
||||||
|
"""
|
||||||
|
{
|
||||||
|
"statusCode": 200,
|
||||||
|
"statusBody": {}
|
||||||
|
}
|
||||||
|
""".trimIndent()
|
||||||
|
)
|
||||||
|
|
||||||
verify(eventPublisher, never()).publishEvent(any())
|
this.kafkaResponseProcessor.onMessage(record)
|
||||||
|
|
||||||
|
verify(eventPublisher, never()).publishEvent(any<ResponseEvent>())
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun shouldNotProcessRecordsWithoutValidBody() {
|
fun shouldProcessRecordsWithAliasNames() {
|
||||||
this.kafkaResponseProcessor.onMessage(createkafkaRecord(requestId = "TestID1234", statusBody = null))
|
val record = ConsumerRecord<String, String>(
|
||||||
|
"test-topic",
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
null,
|
||||||
|
"""
|
||||||
|
{
|
||||||
|
"request_id": "test0123456789",
|
||||||
|
"status_code": 200,
|
||||||
|
"status_body": {}
|
||||||
|
}
|
||||||
|
""".trimIndent()
|
||||||
|
)
|
||||||
|
|
||||||
verify(eventPublisher, never()).publishEvent(any())
|
this.kafkaResponseProcessor.onMessage(record)
|
||||||
|
|
||||||
|
verify(eventPublisher, times(1)).publishEvent(any<ResponseEvent>())
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
fun shouldNotProcessRecordsWithoutValidStatusBody() {
|
||||||
|
this.kafkaResponseProcessor.onMessage(createKafkaRecord(requestId = "TestID1234", statusBody = null))
|
||||||
|
|
||||||
|
verify(eventPublisher, never()).publishEvent(any<ResponseEvent>())
|
||||||
}
|
}
|
||||||
|
|
||||||
@ParameterizedTest
|
@ParameterizedTest
|
||||||
@MethodSource("statusCodeSource")
|
@MethodSource("statusCodeSource")
|
||||||
fun shouldProcessValidRecordsWithStatusCode(statusCode: Int) {
|
fun shouldProcessValidRecordsWithStatusCode(statusCode: Int) {
|
||||||
this.kafkaResponseProcessor.onMessage(createkafkaRecord("TestID1234", statusCode))
|
this.kafkaResponseProcessor.onMessage(createKafkaRecord("TestID1234", statusCode))
|
||||||
verify(eventPublisher, times(1)).publishEvent(any<ResponseEvent>())
|
verify(eventPublisher, times(1)).publishEvent(any<ResponseEvent>())
|
||||||
}
|
}
|
||||||
|
|
||||||
|